Uniapp + SpringBoot разрабатывает проект WeChat H5 и авторизует общедоступную учетную запись WeChat для входа в бэкэнд JAVA (1. Настройте и используйте общедоступную платформу WeChat для тестирования общедоступной учетной записи).

Подайте заявку на тестовую учетную запись для отладки и разработки. Тестовая учетная запись имеет разрешения интерфейса большинства учетных записей служб.

1. Заполните и проверьте информацию о конфигурации интерфейса.

Здесь вам нужно заполнить URL-адрес и строку проверки токена.
Я использовал проникновение в интрасеть natapp, чтобы сопоставить локальную серверную службу порта 8080 с http://x7zws8.natappfree.cc
https://natapp.cn/ на официальном сайте natapp. Зарегистрируйте учетную запись и подайте заявку на бесплатный туннель.
Вставьте сюда описание изображения
После подачи заявки привяжите доменное имя к своему собственному внутреннему
Вставьте сюда описание изображения
интерфейсу:

    @RequestMapping(value = "/check", produces = "text/plain;charset=UTF-8", method = {
    
    RequestMethod.GET, RequestMethod.POST})
    //微信服务器根据配置的token,结合时间戳timestamp和随机数nonce通过SHA1生成签名,发起get请求,检验token的正确性,
    //检验正确原样返回随机字符串echostr,失败返回空字符串
    public String check(HttpServletRequest request, HttpServletResponse response,
                        @RequestParam("signature") String signature,
                        @RequestParam("timestamp") String timestamp,
                        @RequestParam("nonce") String nonce,
                        String echostr) throws Exception {
    
    
        //若是为get请求,则为开发者模式验证
        if ("get".equals(request.getMethod().toLowerCase())) {
    
    
            String checkSignature = SHA1.creatSHA1("token", timestamp, nonce);
            if (checkSignature.equals(signature)) {
    
    
                return echostr;
            }
        }
        return null;
    }

ША1:

import java.security.MessageDigest;
import java.util.Arrays;

//SHA1加密算法类
public class SHA1 {
    
    
    /**
     *
     * @param token
     * @param timestamp 时间戳
     * @param nonce 随机字符串
     * @return 安全签名
     * @throws AesException
     */
    public static String creatSHA1(String token, String timestamp, String nonce) throws AesException
    {
    
    
        try {
    
    
            String[] array = new String[] {
    
     token, timestamp, nonce};
            StringBuffer sb = new StringBuffer();
            // 字符串排序
            Arrays.sort(array);
            for (int i = 0; i < 3; i++) {
    
    
                sb.append(array[i]);
            }
            String str = sb.toString();
            // SHA1签名生成
            MessageDigest md = MessageDigest.getInstance("SHA-1");
            md.update(str.getBytes());
            byte[] digest = md.digest();

            StringBuffer hexstr = new StringBuffer();
            String shaHex = "";
            for (int i = 0; i < digest.length; i++) {
    
    
                shaHex = Integer.toHexString(digest[i] & 0xFF);
                if (shaHex.length() < 2) {
    
    
                    hexstr.append(0);
                }
                hexstr.append(shaHex);
            }
            return hexstr.toString();
        } catch (Exception e) {
    
    
            e.printStackTrace();
            throw new AesException(AesException.ComputeSignatureError);
        }
    }
}

Исключение Aes:

public class AesException extends Exception {
    
    

    public final static int OK = 0;
    public final static int ValidateSignatureError = -40001;
    public final static int ParseXmlError = -40002;
    public final static int ComputeSignatureError = -40003;
    public final static int IllegalAesKey = -40004;
    public final static int ValidateAppidError = -40005;
    public final static int EncryptAESError = -40006;
    public final static int DecryptAESError = -40007;
    public final static int IllegalBuffer = -40008;

    private int code;

    private static String getMessage(int code) {
    
    
        switch (code) {
    
    
            case ValidateSignatureError:
                return "签名验证错误";
            case ParseXmlError:
                return "xml解析失败";
            case ComputeSignatureError:
                return "sha加密生成签名失败";
            case IllegalAesKey:
                return "SymmetricKey非法";
            case ValidateAppidError:
                return "appid校验失败";
            case EncryptAESError:
                return "aes加密失败";
            case DecryptAESError:
                return "aes解密失败";
            case IllegalBuffer:
                return "解密后得到的buffer非法";
            default:
                return null; // cannot be
        }
    }

    public int getCode() {
    
    
        return code;
    }

    public AesException(int code) {
    
    
        super(getMessage(code));
        this.code = code;
    }
}

После завершения проверки настройте авторизацию веб-страницы в службе веб-страниц в тестовой учетной записи, чтобы получить базового пользователя.
Вставьте сюда описание изображения
Вставьте сюда описание изображения

Здесь введите адрес вашего внешнего проекта после запуска. Мой внутренний порт проекта — 8080. После запуска внешнего интерфейса uniapp порт — 8081, поэтому я настроил локальный ipv4 ip + номер порта.

2. Конфигурация бэкенда + Сборка URL + Получение OpenID пользователя и базовой информации через Код

Внедрение зависимостей в pom проекта SpringBoot.

        <dependency>
            <groupId>com.github.binarywang</groupId>
            <artifactId>weixin-java-mp</artifactId>
            <version>4.6.0</version>
        </dependency>

Создать класс конфигурации

import com.ruoyi.system.domain.WechatConfig;
import com.ruoyi.system.service.IWechatConfigService;
import me.chanjar.weixin.mp.api.WxMpService;
import me.chanjar.weixin.mp.api.impl.WxMpServiceImpl;
import me.chanjar.weixin.mp.config.WxMpConfigStorage;
import me.chanjar.weixin.mp.config.impl.WxMpDefaultConfigImpl;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class WxMpConfig {
    
    

    @Autowired
    private IWechatConfigService wechatConfigService;

    @Bean
    public WxMpConfigStorage wxMpConfigStorage() {
    
    
        WxMpDefaultConfigImpl configStorage = new WxMpDefaultConfigImpl();
        WechatConfig wechatConfig = wechatConfigService.selectWechatConfigById(1L);
        //我这里是做成了获取数据库中保存的AppID和AppSercret正常情况下写死就可以
        configStorage.setAppId(wechatConfig.getAppId());
        configStorage.setSecret(wechatConfig.getAppSecret());
        return configStorage;
    }

    @Bean
    public WxMpService wxMpService(WxMpConfigStorage configStorage) {
    
    
        WxMpService wxMpService = new WxMpServiceImpl();
        wxMpService.setWxMpConfigStorage(configStorage);
        return wxMpService;
    }
}

После этого вы можете использовать его в контроллере в обычном режиме.
Интерфейс /authorize фактически выполняет только один шаг: создание URL-адреса для получения кода. После завершения построения он возвращается во внешний интерфейс. Интерфейс напрямую перенаправляет
три параметры buildAuthorizationUrl.
Первый — это адрес обратного вызова.
Второй заполняется. snsapi_base или snsapi_userinfo
snsapi_base может получить openid, тогда как snsapi_userinfo может получить основную информацию о пользователе.
snsapi_base можно получить автоматически, а snsapi_userinfo требует, чтобы пользователь щелкнул для авторизации.

@RestController
@RequestMapping("/wx")
public class WeChatController extends BaseController {
    
    
    @GetMapping("/authorize")
    public AjaxResult test() {
    
    
        WxOAuth2Service oAuth2Service = wxMpService.getOAuth2Service();
        String redirectUrl = oAuth2Service.buildAuthorizationUrl("http://192.168.1.100:8081", "snsapi_userinfo", null);
        return success(redirectUrl);
    }

    @GetMapping("/userInfo")
    public AjaxResult userInfo(@RequestParam("code") String code) throws WxErrorException {
    
    
        WxOAuth2Service oAuth2Service = wxMpService.getOAuth2Service();
        WxOAuth2AccessToken wxMpOAuth2AccessToken = oAuth2Service.getAccessToken(code);
        logger.info("【wxMpOAuth2AccessToken:】{}", wxMpOAuth2AccessToken);
        String openId = wxMpOAuth2AccessToken.getOpenId();
        logger.info("【openid:】{}", openId);
        WxOAuth2UserInfo userInfo = oAuth2Service.getUserInfo(wxMpOAuth2AccessToken, "zh_CN");
        logger.info("【用户信息:】{}", userInfo.toString());
        return success(userInfo);
    }
}

2. Вызов внешнего интерфейса Uniapp

Интерфейсный интерфейс сначала вызывает /authorize, чтобы получить собранный адрес, а затем перенаправляет и получает код
getUrlCode, чтобы определить, содержит ли адрес код. Если да, используйте код, чтобы получить основную информацию пользователя и openid.

<template>
	<view class="content">
		<image class="logo" :src="loginUser.headImgUrl ? loginUser.headImgUrl : '/static/logo.png'"></image>
		<view class="text-area">
			<text class="title">{
    
    {
    
    loginUser.openid}}</text>
		</view>
		<button type="default" @click="login">登录</button>
	</view>
</template>

<script>
	export default {
    
    
		data() {
    
    
			return {
    
    
				loginUser:{
    
    }
			}
		},
		onLoad(query) {
    
    
			const that = this;
			let code = that.getUrlCode();
			if (code) {
    
    
				console.log("有code")
				console.log(code)
				uni.request({
    
    
					url: "http://192.168.1.100:8080/wx/userInfo",
					data: {
    
    
						code: code
					},
					success(res) {
    
    
						console.log("获取到用户信息")
						console.log(res.data.data);
						that.loginUser = res.data.data;
					}
				})
			} else {
    
    
				console.log("没有code")
			}
		},
		methods: {
    
    
			login() {
    
    
				uni.request({
    
    
					url: "http://192.168.1.100:8080/wx/authorize",
					success(res) {
    
    
						window.location.href = res.data.msg;
					}
				})
			},
			getUrlCode() {
    
    
				return (
					decodeURIComponent(
						(new RegExp("[?|&]" + "code" + "=" + "([^&;]+?)(&|#|;|$)").exec(
							location.href
						) || [, ""])[1].replace(/\+/g, "%20")
					) || null
				);
			},
		}
	}
</script>

рекомендация

отblog.csdn.net/qq_43193513/article/details/136558126