spring native 初體驗實現 小米控制美的空調

2022-09-13 06:19:22

目前關於 spring native 分享的文章還比較少

寫這篇文章的主要目前是分享一下自己寫的一個 小米控制美的空調 的程式 整合 spring native 過程中碰到的一些問題和解決方法

先放地址 : https://github.com/toohandsome/xiaomi2meidi 歡迎star

對比一下速度:

上面是編譯成exe執行,下面是jar執行 快了10倍.


Spring Native 可以通過 GraalVM 將 Spring 應用程式編譯成原生映象,提供了一種新的方式來部署 Spring 應用。

ps: 這篇文章主要是將其打包成exe,沒有打包成docker映象

注意 目前 Spring Native 已經不支援jdk8 了,這裡選用的jvm 是 graalvm-ce-java17-22.2.0, maven 選用 apache-maven-3.8.6

首先在 start.spring.io 中 選擇 spring native 和 web

下載後匯入idea,把專案的sdk 和 語法 均設定為 17

按照 Visual Studio ,我這裡是vs2019 , 更高版本應該也可以,可以參考這篇文章 https://www.cnblogs.com/luguojun/p/16132521.html

環境設定好了以後,

在 resource 下 建立 META-INF/native-image/{groupId}/{artifactId}
然後在下面建立

native-image.properties
proxy-config.json
reflect-config.json
resource-config.json
serialization-config.json

如圖所示

開始編譯主要有幾種錯誤

1. must not contain "."

must not contain ".". This can happen implicitly if the builder runs exclusively on the --module-path but specifies the com.oracle.svm.hosted.NativeImageGeneratorRunner main class without --module.

經過排除發現是 classpath 環境變數不能有 "." , 只要保留

%JAVA_HOME%\lib;

即可,如圖

2.UnsupportedFeatureError

UnsupportedFeatureError: Proxy class defined by interfaces[xxxx] not found. Generating proxy classes at runtime is not supported. Proxy classes need to be defined at image build time by specifying the list of interfaces that they implement. 

會一直報錯 某某bean 不能被建立 , 需要在 proxy-config.json 中 增加 報錯資訊中的 xxx 介面.
但是這裡有個問題就是 它這個錯會有很多,你改了一個bean , 下一個bean 又會有不同的介面,所以我寫了一個程式來自動分析

      Pattern p = Pattern.compile("Proxy class defined by interfaces \\[(.*?)\\]");
       for (int i = 0; i < 10000; i++) {
            Process process = Runtime.getRuntime().exec("F:\\springnative\\start.bat");
            BufferedReader bufferedReader = new BufferedReader(
                    new InputStreamReader(process.getInputStream(), "gbk"));
            String line;
            while ((line = bufferedReader.readLine()) != null) {
                System.out.println(line);
            }
            System.out.println("success");

            String s = RuntimeUtil.execForStr("\"F:\\springnative\\target\\xiaomi2meidi.exe\"");
            System.out.println("exe: " + s);
            Matcher matcher = p.matcher(s);
            if (matcher.find()) {
                var interfaces = matcher.group(1).replace("interface ", "").replace(" ", "");
                
                String[] split = interfaces.split(",");
                JSONArray jsonArray = new JSONArray();
                for (String s1 : split) {
                    System.out.println(s1);
                    jsonArray.add(s1);
                }
                String s1 = Files.readString(Paths.get("F:\\springnative\\src\\main\\resources\\META-INF\\native-image\\com.yxd.xiaomi2meidi\\proxy-config.json"));
                List<JSONArray> jsonArrays = JSON.parseArray(s1, JSONArray.class);
                jsonArrays.add(jsonArray);
                String pretty = JSON.toJSONString(jsonArrays, JSONWriter.Feature.PrettyFormat,
                        JSONWriter.Feature.WriteMapNullValue,
                        JSONWriter.Feature.WriteNullListAsEmpty);
                try {
                    Files.writeString(Paths.get("F:\\springnative\\src\\main\\resources\\META-INF\\native-image\\com.yxd.xiaomi2meidi\\proxy-config.json"), pretty);
                } catch (Exception e) {
                    e.printStackTrace();
                }

            } else {
                break;
            }

        }

寫的比較粗糙
大概就是迴圈編譯執行,把執行後得到的紀錄檔 用正則匹配出來,然後自動加到 proxy-config.json 中去,然後又重新編譯,直到它不報錯為止.

3. logback 沒有紀錄檔或者報錯找不到 ConsoleAppender 等紀錄檔相關的類

增加依賴

<dependency>
  <groupId>org.codehaus.janino</groupId>
  <artifactId>janino</artifactId>
  <version>3.1.8</version>
</dependency>

在 reflect-config.json 增加設定

{
    "name": "ch.qos.logback.core.ConsoleAppender",
    "allPublicConstructors": true,
    "allPublicMethods": true
  }

這裡的類名是怎麼來的呢,其實是 logback-spring.xml 裡面的類, 還有

ch.qos.logback.core.rolling.RollingFileAppender
ch.qos.logback.classic.PatternLayout
ch.qos.logback.classic.encoder.PatternLayoutEncoder
ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy

等, 根據自己的xml設定


最後說一下 如何使用 小米控制美的空調

首先說一下思路.

想用小愛控制,那麼最好的辦法自然是接入小米的iot平臺,然而小米並不對個人開發者開發.

退而求其次,我們可以找一個第三方廠商,由他們做中間人接入.

blinker (點燈科技) 是一家物聯網技術提供商, 官網 點燈科技 (https://diandeng.tech/home) (雖然檔案爛,但是功能不含糊,該有的都有. )

雖然我之前參考 使用ESP32和Blinker實現遠端網路喚醒電腦(接入語音助手,以小愛同學為例) 這篇貼文用esp32成功控制了電腦遠端開機

但是這次我不想依賴硬體裝置,不然成本太高了( 一臺空調裝置就得買一個esp32),

然後我就想blinker的程式碼既然能在樹莓派上面跑,自然我就能把核心邏輯摳出來用java重寫一遍

好.正文開始

首先我們在blinker官網下載 他們的app,註冊.

註冊後 登入,顯示如下介面


你們這裡應該是空的,我加過所以有其他裝置

然後我們點選右上角的 加號 進行新增一個新的裝置

點選 獨立裝置

選擇網路接入

得到 authKey, 儲存好,後面要用

然後我們返回裝置列表,點選剛新加的裝置. 右上角 三個點

編輯裝置名稱

輸入名稱, 這個名稱就是後面你喊小愛的名稱,同時 要和 美的美居 app 裡面空調的名稱要相同

確認修改後,我們下載 github 上的 程式 ,執行後在 程式提示的組態檔中輸入

phone: 美的app手機號, password: 美的app密碼,acNameList: 空調名稱(多個用逗號隔開), blinkerKeyList: 點燈的authkey(多個用逗號隔開,需要與空調名稱一一對應)

{
        "phone":"13812345678",
        "password":"123456",
        "acNameList":"書房空調,主臥空調,次臥空調",
        "blinkerKeyList":"8*****2,2*****9,0******8",
        "uid":"",
        "accessToken":"",
        "tokenPwd":"",
        "homeId":"",
        "appVersion":"",
        "deviceId":"",
        "deviceName":"",
        "osVersion":"",
        "deviceList":[]
}

程式正常執行後.

我們 開啟 米家 app , 點選 "我的" , 往下翻 選擇 "其他平臺裝置"

先點新增, 找到 點燈科技. ,然後點選 同步裝置.


如果這裡出現了你剛新加的裝置說明就成功了.

然後就可以用小愛控制了