JLChen
2021-01-05 f500e14c0a994487070380c50c85e0929cbc8e63
Crabtree/SmartHome/UI/SimpleControl/MqttCommon.cs
@@ -17,10 +17,16 @@
{
    public static class MqttCommon
    {
        /// <summary>
        /// 是否提示 远程连接的错误信息
        /// </summary>
        public static bool IfDEBUG = false;
        static string mqttEncryptKey = "";
        static string checkGatewayTopicBase64 = "";
        static RemoteMACInfo CurRemoteMACInfo = null;
        static MqttInfo mMqttInfo = null;
        public static bool IsGatewayOnline = true;
        /// <summary>
        /// 手机标识
@@ -44,6 +50,7 @@
        static MqttCommon ()
        {
            InitMqtt ();
            //InitCheckGateway ();
        }
        /// <summary>
@@ -54,15 +61,28 @@
        {
            new System.Threading.Thread (async () => {
                while (true) {
                    System.Threading.Thread.Sleep (100);
                    if (!CommonPage.IsRemote) continue;
                    try {
                        System.Threading.Thread.Sleep (200);
                        if (!CommonPage.IsRemote) continue;
                    await StartCloudMqtt ();
                    await CheckingSubscribeTopics ();
                        if (Control.IsEnterBackground) continue;
                        await StartCloudMqtt ();
                        await CheckingSubscribeTopics ();
                    } catch { }
                }
            }) { IsBackground = true }.Start ();
        }
        public static void InitState ()
        {
            IfNeedReadAllDeviceStatus = true;
            bNeedStartTip = true;
            IsGatewayOnline = false;
            StartCloudMqtt ();
        }
        /// <summary>
        /// 是否订阅成功
@@ -79,6 +99,7 @@
            }
            try {
                if (!isSubscribeTopicSuccess) {
                    var topicFilterCommon = new TopicFilter () {
                        Topic = $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/Common/#",
@@ -97,34 +118,64 @@
                        QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce
                    };
                    var topicFilters = new TopicFilter [] { topicFilterCommon, topicFilterGateWayInfoChange, topicFilterNotifySqueeze };
                    //网关掉线主题
                    var topicFilterNotifyGateWayOffline = new TopicFilter () {
                        Topic = $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/NotifyGateWayOffline",
                        QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce
                    };
                    var topicFilters = new TopicFilter [] { topicFilterCommon, topicFilterGateWayInfoChange, topicFilterNotifySqueeze, topicFilterNotifyGateWayOffline };
                    var result = await RemoteMqttClient.SubscribeAsync (topicFilters);
                    if (result.Items [0].ResultCode == MQTTnet.Client.Subscribing.MqttClientSubscribeResultCode.GrantedQoS2) {
                        isSubscribeTopicSuccess = true;
                    }
                    ////2020-06-22 通用主题
                    //var topicFilterMacMark = new TopicFilter () {
                    //    Topic = $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/#",
                    //    QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce
                    //};
                    ////挤下线主题
                    //var topicFilterNotifySqueeze = new TopicFilter () {
                    //    Topic = $"/BusGateWayToClient/{mMqttInfo.connEmqClientId}/Push/NotifySqueeze",
                    //    QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce
                    //};
                    //var topicFilters = new TopicFilter [] { topicFilterMacMark, topicFilterNotifySqueeze };
                    //var result = await RemoteMqttClient.SubscribeAsync (topicFilters);
                    //if (result.Items [0].ResultCode == MQTTnet.Client.Subscribing.MqttClientSubscribeResultCode.GrantedQoS2) {
                    //    isSubscribeTopicSuccess = true;
                    //}
                }
            } catch (Exception e) {
            } catch (Exception ex) {
            }
        }
        /// <summary>
        /// 断开远程Mqtt的链接
        /// <summary>
        /// 断开远程Mqtt的链接
        /// </summary>
        public static async System.Threading.Tasks.Task DisConnectRemoteMqttClient (string s = "")
        {
            try {
                if (remoteIsConnected) {
                    remoteIsConnected = false;
                    System.Console.WriteLine ($"============>MqttRemote主动断开_{s}");
                    //await RemoteMqttClient.DisconnectAsync(new MQTTnet.Client.Disconnecting.MqttClientDisconnectOptions { }, CancellationToken.None);
        public static async System.Threading.Tasks.Task DisConnectRemoteMqttClient (string s = "")
        {
            try {
                if (remoteIsConnected) {
                    remoteIsConnected = false;
                    Utlis.WriteLine ($"============>MqttRemote主动断开_{s}");
                    //await RemoteMqttClient.DisconnectAsync(new MQTTnet.Client.Disconnecting.MqttClientDisconnectOptions { }, CancellationToken.None);
                    await RemoteMqttClient.DisconnectAsync ();
                }
                    if (CommonPage.IsRemote) {
                        Utlis.ShowAppLinkStatus (AppLinkStatus.CloudUnlink);
                    }
                    //Utlis.WriteLine ($"============>MqttRemoteDisConnectRemoteMqttClient");
                }
            } catch (Exception e) {
                System.Console.WriteLine ($"============>MqttRemote断开通讯连接出异常:{e.Message}");
            }
                Utlis.WriteLine ($"============>MqttRemote断开通讯连接出异常:{e.Message}");
            }
        }
        static DateTime dateTime = DateTime.MinValue;
@@ -134,13 +185,10 @@
        static string PushSignStr = System.DateTime.Now.Ticks.ToString ();
        /// <summary>
        /// 外网的MQTT是否正在连接
        /// </summary>
        static bool remoteMqttIsConnecting;
        public static bool remoteMqttIsConnecting;
        static bool remoteIsConnected;
        /// <summary>
        /// 启动远程Mqtt
@@ -160,180 +208,210 @@
            await System.Threading.Tasks.Task.Factory.StartNew (async () => {
                try {
                    //lock (RemoteMqttClient) {
                        //表示后面将进行连接
                    //表示后面将进行连接
                        #region 初始化远程Mqtt
                        //(3)当[连接云端的Mqtt成功后]或者[以及后面App通过云端Mqtt转发数据给网关成功后],处理接收到云端数据包响应时在mqttServerClient_ApplicationMessageReceived这个方法处理
                        if (RemoteMqttClient.ApplicationMessageReceivedHandler == null) {
                            RemoteMqttClient.UseApplicationMessageReceivedHandler (async (e) => {
                                try {
                                    if (!RemoteMqttClient.IsConnected || !CommonPage.IsRemote) {
                                        return;
                    #region 初始化远程Mqtt
                    //2020-06-22 增加
                    RemoteMqttClient = new MqttFactory ().CreateMqttClient ();
                    //(3)当[连接云端的Mqtt成功后]或者[以及后面App通过云端Mqtt转发数据给网关成功后],处理接收到云端数据包响应时在mqttServerClient_ApplicationMessageReceived这个方法处理
                    if (RemoteMqttClient.ApplicationMessageReceivedHandler == null) {
                        RemoteMqttClient.UseApplicationMessageReceivedHandler (async (e) => {
                            try {
                                if (!RemoteMqttClient.IsConnected || !CommonPage.IsRemote) {
                                    return;
                                }
                                var aesDecryptTopic = e.ApplicationMessage.Topic;
                                var aesDecryptPayload = e.ApplicationMessage.Payload;
                                //Utlis.WriteLine ("Topic={0}", aesDecryptTopic);
                                //if (aesDecryptTopic == $"NotifyBusGateWayInfoChagne/{CurRemoteMACInfo.md5_mac_string}") {//网关上线,需要更新aeskey
                                //2020-01-11 修改订阅主题地址
                                if (aesDecryptTopic == $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/NotifyBusGateWayInfoChange") {//网关上线,需要更新aeskey                                                                                                                     //----第二步:读取账号下面的网关列表
                                    await ReceiveNotifyBusGateWayInfoChange ();
                                } else if (aesDecryptTopic == $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/NotifyGateWayOffline") {//网关掉线                                                                                                                    //----第二步:读取账号下面的网关列表
                                    ReceiveNotifyGateWayOffline ();
                                } else if (aesDecryptTopic == $"/BusGateWayToClient/{mMqttInfo.connEmqClientId}/Push/NotifySqueeze") {//订阅挤下线问题
                                    await ReceiveNotifySqueezeAsync (aesDecryptPayload);
                                } else if (aesDecryptTopic == $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/Common/CheckGateway") {
                                    ReceiveCheckGatewayTopic ();
                                } else {
                                    SetGatewayOnlineResetCheck ();
                                    if (!string.IsNullOrEmpty (mqttEncryptKey)) {
                                        aesDecryptPayload = Shared.Securitys.EncryptionService.AesDecryptPayload (e.ApplicationMessage.Payload, mqttEncryptKey);
                                    }
                                    var aesDecryptTopic = e.ApplicationMessage.Topic;
                                    var aesDecryptPayload = e.ApplicationMessage.Payload;
                                    //Console.WriteLine ("Topic={0}", aesDecryptTopic);
                                    //if (aesDecryptTopic == $"NotifyBusGateWayInfoChagne/{CurRemoteMACInfo.md5_mac_string}") {//网关上线,需要更新aeskey
                                    //2020-01-11 修改订阅主题地址
                                    if (aesDecryptTopic == $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/NotifyBusGateWayInfoChange") {//网关上线,需要更新aeskey                                                                                                                     //----第二步:读取账号下面的网关列表
                                        await ReceiveNotifyBusGateWayInfoChange ();
                                    } else if (aesDecryptTopic == $"/BusGateWayToClient/{mMqttInfo.connEmqClientId}/Push/NotifySqueeze") {//订阅挤下线问题
                                        await ReceiveNotifySqueezeAsync (aesDecryptPayload);
                                    }else if (aesDecryptTopic == $"/BusGateWayToClient/{CurRemoteMACInfo.macMark}/Common/CheckGateway") {
                                        MainPage.WiFiStatus = "CrabtreeAdd/CloudUnlink.png";
                                        // = $"/ClientToBusGateWay/{CurRemoteMACInfo.macMark}/Common/OldON";
                                        var ss = CommonPage.MyEncodingUTF8.GetString (aesDecryptPayload);
                                        var obj = Newtonsoft.Json.JsonConvert.DeserializeObject<ResponsePack> (ss);
                                        if (obj == null) {
                                            return;
                                        }
                                        switch (obj.StateCode) {
                                        case "HDLUdpDataForwardServerMqttClientNoOnLine":
                                        case "NoOnline":
                                        case "NetworkAnomaly"://不在线
                                            MainPage.AddTip ("Gateway offline");
                                            //Application.RunOnMainThread (() => {
                                            //    Shared.SimpleControl.Phone.UserMiddle.LinkStatusTip.BackgroundColor = SkinStyle.Current.DelColor;
                                            //});
                                            break;
                                        case "NoRecord"://MAC不正确
                                            MainPage.AddTip (Language.StringByID (Shared.SimpleControl.R.MyInternationalizationString.MACError));
                                            //Application.RunOnMainThread (() => {
                                            //    Shared.SimpleControl.Phone.UserMiddle.LinkStatusTip.BackgroundColor = SkinStyle.Current.DelColor;
                                            //});
                                            break;
                                        case "Success":
                                            MainPage.AddTip (UserConfig.Instance.CurrentRegion.Name + ":" + Language.StringByID (Shared.SimpleControl.R.MyInternationalizationString.LinkSuccess));
                                            MainPage.WiFiStatus = "CrabtreeAdd/CloudLink.png";
                                            break;
                                        default:
                                            MainPage.AddTip (Language.StringByID (Shared.SimpleControl.R.MyInternationalizationString.LinkLoser));
                                            //Application.RunOnMainThread (() => {
                                            //    Shared.SimpleControl.Phone.UserMiddle.LinkStatusTip.BackgroundColor = SkinStyle.Current.DelColor;
                                            //});
                                            break;
                                        }
                                        Application.RunOnMainThread (() => {
                                            UserMiddle.btnLinkStatus.UnSelectedImagePath = MainPage.WiFiStatus;
                                        });
                                    } else {
                                        if (!string.IsNullOrEmpty (mqttEncryptKey)) {
                                            aesDecryptPayload = Shared.Securitys.EncryptionService.AesDecryptPayload (e.ApplicationMessage.Payload, mqttEncryptKey);
                                        }
                                    }
                                    var packet = new Packet ();
                                    packet.IsLocal = false;
                                    packet.Bytes = aesDecryptPayload;
                                    packet.Manager ();
                                } catch { }
                            });
                        }
                        if (RemoteMqttClient.DisconnectedHandler == null) {
                            RemoteMqttClient.UseDisconnectedHandler (async (e) => {
                                System.Console.WriteLine ($"============>Mqtt远程连接断开");
                                isSubscribeTopicSuccess = false;
                                await DisConnectRemoteMqttClient ("StartRemoteMqtt.DisconnectedHandler");
                                //await StartRemoteMqtt();
                                //if (thisShowTip) {
                                //    if (CommonPage.IsRemote) {
                                //        Application.RunOnMainThread (() => {
                                //            MainPage.Loading.Hide ();
                                //        });
                                //    }
                                //} else {
                                //    thisShowTip = true;
                                //}
                            });
                        }
                        if (RemoteMqttClient.ConnectedHandler == null) {
                            RemoteMqttClient.UseConnectedHandler (async (e) => {
                                Shared.SimpleControl.Phone.UserMiddle.ReadAllDeviceStatus ();
                                System.Console.WriteLine ($"============>Mqtt远程连接成功");
                                if (CommonPage.IsRemote) {
                                    Utlis.ShowAppLinkStatus (AppLinkStatus.CloudLink);
                                }
                                if (CurRemoteMACInfo != null) {
                                    if (CurRemoteMACInfo.isValid == "InValid") {
                                        MainPage.AddTip ("Remote failed,gateway offline");
                                    } else {
                                        MqttRemoteSend (new byte [] { 0 }, 3);
                            } catch { }
                        });
                    }
                    if (RemoteMqttClient.DisconnectedHandler == null) {
                        RemoteMqttClient.UseDisconnectedHandler (async (e) => {
                            Utlis.WriteLine ($"============>Mqtt远程连接断开");
                            isSubscribeTopicSuccess = false;
                            await DisConnectRemoteMqttClient ("StartRemoteMqtt.DisconnectedHandler");
                        });
                    }
                    if (RemoteMqttClient.ConnectedHandler == null) {
                        RemoteMqttClient.UseConnectedHandler (async (e) => {
                            //IfNeedReadAllDeviceStatus = true;
                            mCheckGatewayTime = DateTime.Now;
                            CheckGatewayCount = 0;
                            bNeedStartTip = true;
                            Utlis.WriteLine ($"============>Mqtt远程连接成功");
                            if (CommonPage.IsRemote) {
                                Utlis.ShowAppLinkStatus (AppLinkStatus.CloudLink);
                            }
                            if (CurRemoteMACInfo != null) {
                                IsGatewayOnline = CurRemoteMACInfo.isValid != "InValid";
                                if (!IsGatewayOnline) {
                                    //网关不在线
                                    if (CommonPage.IsRemote) {
                                        Utlis.ShowAppLinkStatus (AppLinkStatus.CloudOffline);
                                    }
                                    MainPage.AddTip ("Remote failed,gateway offline");
                                } else {
                                    //Remote Connection Succeeded
                                    //网关在线
                                    //重新一次所有设备状态
                                    IfNeedReadAllDeviceStatus = false;
                                    Shared.SimpleControl.Phone.UserMiddle.ReadAllDeviceStatus ();
                                    //
                                    MqttRemoteSend (new byte [] { 0 }, 3);
                                    if (CommonPage.IsRemote) {
                                        MainPage.AddTip (Language.StringByID (Shared.SimpleControl.R.MyInternationalizationString.LinkSuccess));
                                    }
                                }
                            });
                        }
                        #endregion
                            }
                        });
                    }
                    #endregion
                    //}
                        mqttRequestParToken = MainPage.LoginUser.LoginTokenString;
                        //--第一步:获取mqtt链接参数
                        var mqttInfoRequestResult = new ResponsePack () { StateCode = "" };
                        //--判断是当前是否分享的住宅
                        if (!UserConfig.Instance.CurrentRegion.IsOthreShare) {
                            var mqttInfoUrl = $"{MainPage.RequestHttpsHost}/EmqMqtt/GetConnMqttInfo";//获取连接远程云端Emq Mqtt 服务器连接信息
                            var mqttInfoRequestPar = new GetConnMqttInfoObj () {
                                LoginAccessToken = mqttRequestParToken,
                                PlatformStr = "L1",
                                PublishPayloadJsonStr = PushSignStr,
                            };
                            mqttInfoRequestResult = MainPage.RequestHttps ("", Newtonsoft.Json.JsonConvert.SerializeObject (mqttInfoRequestPar), mqttInfoUrl);
                            if (mqttInfoRequestResult != null && mqttInfoRequestResult.ResponseData != null) {
                                var mqttInfoRequestResult_info = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttInfo> (mqttInfoRequestResult.ResponseData.ToString ());
                                if (mqttInfoRequestResult_info != null) {
                                    mMqttInfo = mqttInfoRequestResult_info;
                                    //----第二步:读取账号下面的网关列表
                                    var gatewayListUrl = $"{MainPage.RequestHttpsHostMqtt}/Center/Center/GetGatewayPagger"; //App、Buspro软件登录后获取网关列表 http 请求
                                    var gatewayListRequestPar = new RemoteRequestParameters () { LoginAccessToken = mqttRequestParToken, RequestVersion = "RequestVersion1", RequestProtocolType = 0, RequestSource = 1 };
                                    var gatewayListRequestResult = MainPage.RequestHttps ("", Newtonsoft.Json.JsonConvert.SerializeObject (gatewayListRequestPar), gatewayListUrl);
                                    var gatewayListRequestResult_Obj = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttRemoteInfo> (gatewayListRequestResult.ResponseData.ToString ());
                                    //--找出是否存在匹配当前住宅的mac,存在再进行远程。
                                    if (UserConfig.Instance.CheckHomeGatewaysNotEmpty ()) {
                                        CurRemoteMACInfo = gatewayListRequestResult_Obj.pageData.Find ((obj) => obj.mac == UserConfig.Instance.CurrentRegion.HomeGateways [0].GatewayUniqueId);
                                        //CurRemoteMACInfo = gatewayListRequestResult_Obj.pageData [0];
                                        await MQTTConnectAsync ();
                                    } else {
                                        System.Console.WriteLine ("============>Mqtt 取消连接,当前住宅没绑定网关");
                                    }
                                }
                            }
                        } else {
                            //如果是分享过来的住宅 走下面流程
                            var mqttInfoRequestPar = new ShareMemberConnMqttInfoObj () {
                                LoginAccessToken = mqttRequestParToken,
                                PlatformStr = "L1",
                                PublishPayloadJsonStr = PushSignStr,
                                MainUserDistributedMark = UserConfig.Instance.CurrentRegion.MainUserDistributedMark,
                                HomeId = UserConfig.Instance.CurrentRegion.Id
                            };
                            mqttInfoRequestResult = MainPage.RequestHttps (API.ShareMemberConnMqttInfo, Newtonsoft.Json.JsonConvert.SerializeObject (mqttInfoRequestPar));
                            //--第一步:获取mqtt链接参数
                            if (mqttInfoRequestResult != null && mqttInfoRequestResult.ResponseData != null) {
                                var mqttInfoRequestResult_info = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttInfo> (mqttInfoRequestResult.ResponseData.ToString ());
                                if (mqttInfoRequestResult_info != null) {
                                    mMqttInfo = mqttInfoRequestResult_info;
                                    //--第二步:获取当前住分享宅网关信息并连接MQTT
                                    await GetSingleHomeGatewayPaggerAndMQTTConnectAsync ();
                                }
                            }
                        }
                    //提示正在获取连接参数..."
                    ShowStartTip ();
                    mqttRequestParToken = MainPage.LoginUser.LoginTokenString;
                    //--判断是当前是否分享的住宅
                    if (!UserConfig.Instance.CurrentRegion.IsOthreShare) {
                        //主账号获取MQTT 远程链接信息,并连接
                        await GetMqttInfoAndMQTTConnectAsync ();
                    } else {
                        //如果是分享过来的住宅 走下面流程
                        await GetShareMqttInfoAndMQTTConnectAsync ();
                    }
                } catch (Exception ex) {
                    System.Console.WriteLine ($"============>Mqtt 远程连接通讯连接出异常:{ex.Message}");
                    //Utlis.WriteLine ($"============>Mqtt 远程连接通讯连接出异常:{ex.Message}");
                    MainPage.LoadingTipShow ("Connect error, trying to reconnect.");
                    if (IfDEBUG) {
                        MainPage.ShowAlertOnMainThread ("error: " + ex.Message);
                    }
                } finally {
                    //最终要释放连接状态
                    remoteMqttIsConnecting = false;
                    MainPage.LoadingTipHide ();
                    //MainPage.LoadingHide ();
                }
            });
        }
        /// <summary>
        /// 主账号获取MQTT 远程链接信息,并连接
        /// </summary>
        /// <returns></returns>
        static async Task GetMqttInfoAndMQTTConnectAsync ()
        {
            var mqttInfoRequestPar = new GetConnMqttInfoObj () {
                LoginAccessToken = MainPage.LoginUser.LoginTokenString,
                PlatformStr = "L1",
                PublishPayloadJsonStr = PushSignStr,
                Mac = UserConfig.Instance.GatewayMAC,
            };
            //----第一步:获取mqtt链接参数
            var mqttInfoRequestResult = MainPage.RequestHttps (API.GetConnMqttInfo, Newtonsoft.Json.JsonConvert.SerializeObject (mqttInfoRequestPar));
            if (mqttInfoRequestResult != null && mqttInfoRequestResult.ResponseData != null) {
                try {
                    var mqttInfoRequestResult_Obj = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttInfo> (mqttInfoRequestResult.ResponseData.ToString ());
                    if (mqttInfoRequestResult_Obj != null) {
                        mMqttInfo = mqttInfoRequestResult_Obj;
                        string url = mqttInfoRequestResult_Obj.connEmqDomainPort;
                        string clientId = mqttInfoRequestResult_Obj.connEmqClientId;
                        string username = mqttInfoRequestResult_Obj.connEmqUserName;
                        string passwordRemote = mqttInfoRequestResult_Obj.connEmqPwd;
                        if (mqttInfoRequestResult_Obj.AccountAllGateways != null && mqttInfoRequestResult_Obj.AccountAllGateways.Count > 0) {
                            //----第二步找出是否存在匹配当前住宅的mac,存在再进行远程。
                            CurRemoteMACInfo = mqttInfoRequestResult_Obj.AccountAllGateways.Find ((obj) => obj.mac == UserConfig.Instance.GatewayMAC);
                            if (CurRemoteMACInfo != null) {
                                CurRemoteMACInfo.LoginAccessToken = MainPage.LoginUser.LoginTokenString;
                                CurRemoteMACInfo.clientId = clientId;
                                //----第三步 开始连接
                                await MQTTConnectAsync ();
                            } else {
                                //Utlis.WriteLine ("============>Mqtt CurRemoteMACInfo null");
                                MainPage.LoadingTipShow ("Please wait, failed to get gateway parameters.");
                            }
                        }
                    } else {
                        //Utlis.WriteLine ("============>Mqtt mqttInfoRequestResult_Obj null");
                        MainPage.LoadingTipShow ("Please wait, failed to get user parameters.");
                        //MainPage.LoadingTipShow ("Please wait, failed to get parameters.");
                    }
                } catch {
                }
            } else {
                //Utlis.WriteLine ("============>Mqtt GetInfo 失败");
                MainPage.AddTip ("Please wait, failed to get parameters.");
                //MainPage.LoadingTipShow ("Please wait, failed to get parameters.");
            }
        }
        /// <summary>
        /// 子账号获取MQTT 远程链接信息,并连接
        /// </summary>
        /// <returns></returns>
        static async Task GetShareMqttInfoAndMQTTConnectAsync ()
        {
            var mqttInfoRequestPar = new ShareMemberConnMqttInfoObj () {
                LoginAccessToken = mqttRequestParToken,
                PlatformStr = "L1",
                PublishPayloadJsonStr = PushSignStr,
                MainUserDistributedMark = UserConfig.Instance.CurrentRegion.MainUserDistributedMark,
                HomeId = UserConfig.Instance.CurrentRegion.Id
            };
            var mqttInfoRequestResult = MainPage.RequestHttps (API.ShareMemberConnMqttInfo, Newtonsoft.Json.JsonConvert.SerializeObject (mqttInfoRequestPar));
            //--第一步:获取mqtt链接参数
            if (mqttInfoRequestResult != null && mqttInfoRequestResult.ResponseData != null) {
                var mqttInfoRequestResult_info = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttInfo> (mqttInfoRequestResult.ResponseData.ToString ());
                if (mqttInfoRequestResult_info != null) {
                    mMqttInfo = mqttInfoRequestResult_info;
                    //--第二步:获取当前住分享宅网关信息并连接MQTT
                    await GetSingleHomeGatewayPaggerAndMQTTConnectAsync ();
                } else {
                    MainPage.LoadingTipShow ("Please wait, failed to get user parameters.");
                }
            }
        }
        /// <summary>
@@ -341,94 +419,153 @@
        /// </summary>
        static async Task MQTTConnectAsync ()
        {
            if (CurRemoteMACInfo != null && mMqttInfo != null) {
                var url = mMqttInfo.connEmqDomainPort;
                var clientId = mMqttInfo.connEmqClientId;
                var username = mMqttInfo.connEmqUserName;
                var passwordRemote = mMqttInfo.connEmqPwd;
                CurRemoteMACInfo.LoginAccessToken = MainPage.LoginUser.LoginTokenString;
                mqttEncryptKey = CurRemoteMACInfo.aesKey;
                var options1 = new MQTTnet.Client.Options.MqttClientOptionsBuilder ()
                                    .WithClientId (clientId)
                                    .WithTcpServer (url.Split (':') [1].Substring ("//".Length), int.Parse (url.Split (':') [2]))
                                    .WithCredentials (username, passwordRemote)
                                    .WithCleanSession ()
                                   // .WithCommunicationTimeout (new TimeSpan (0, 0, 20))
                                    .Build ();
                //获取参数成功,开始连接...
                //MainPage.AddTip ("Successfully obtained parameters, starting to connect...");
                await DisConnectRemoteMqttClient ("StartRemoteMqtt");
                await RemoteMqttClient.ConnectAsync (options1);
                remoteIsConnected = true;
                //await MqttRemoteSend (new byte [] { 0 }, 1);
                //await MqttRemoteSend (new byte [] { 0 }, 2);
                //await MqttRemoteSend (new byte [] { 0 }, 4);
                MainPage.LoadingTipShow ("Please wait, connecting remotely...");
                //try {
                //    var url = mMqttInfo.connEmqDomainPort;
                //    var clientId = mMqttInfo.connEmqClientId;
                //    var username = mMqttInfo.connEmqUserName;
                //    var passwordRemote = mMqttInfo.connEmqPwd;
                //    CurRemoteMACInfo.clientId = clientId;
                //    CurRemoteMACInfo.LoginAccessToken = MainPage.LoginUser.LoginTokenString;
                //    mqttEncryptKey = CurRemoteMACInfo.isNewBusproGateway ? CurRemoteMACInfo.aesKey : "";
                //    //mqttEncryptKey = CurRemoteMACInfo.aesKey;
                //    var options1 = new MQTTnet.Client.Options.MqttClientOptionsBuilder ()
                //                        .WithClientId (clientId)
                //                        .WithTcpServer (url.Split (':') [1].Substring ("//".Length), int.Parse (url.Split (':') [2]))
                //                        .WithCredentials (username, passwordRemote)
                //                        .WithCleanSession ()
                //                        // .WithCommunicationTimeout (new TimeSpan (0, 0, 20))
                //                        .Build ();
                //    await DisConnectRemoteMqttClient ("StartRemoteMqtt");
                //    await RemoteMqttClient.ConnectAsync (options1);
                //    remoteIsConnected = true;
                //    IsDisConnectingWithSendCatch = false;
                //} catch (Exception ex) {
                //    Console.WriteLine ("Connect error: " + ex.Message);
                //}
                //string mes = "";
                try {
                    var url = mMqttInfo.connEmqDomainPort;
                    var clientId = mMqttInfo.connEmqClientId;
                    var username = mMqttInfo.connEmqUserName;
                    var passwordRemote = mMqttInfo.connEmqPwd;
                    //mes = "step1.Set Data";
                    CurRemoteMACInfo.LoginAccessToken = MainPage.LoginUser.LoginTokenString;
                    mqttEncryptKey = CurRemoteMACInfo.aesKey;
                    var options1 = new MQTTnet.Client.Options.MqttClientOptionsBuilder ()
                                        .WithClientId (clientId)
                                        .WithTcpServer (url.Split (':') [1].Substring ("//".Length), int.Parse (url.Split (':') [2]))
                                        .WithCredentials (username, passwordRemote)
                                        .WithCleanSession ()
                                        // .WithCommunicationTimeout (new TimeSpan (0, 0, 20))
                                        .Build ();
                    //mes = "step2.DisConnect";
                    await DisConnectRemoteMqttClient ("StartRemoteMqtt");
                    //MainPage.AddTip ("Start to connect...");
                    //mes = "step3.Start to connect";
                    await RemoteMqttClient.ConnectAsync (options1);
                    //mes = "step4.Connected Success";
                    remoteIsConnected = true;
                    IsDisConnectingWithSendCatch = false;
                    MainPage.LoadingTipShow ("Remote connection succeeded.");
                    //MainPage.AddTip ("Connection succeeded!");
                    //await MqttRemoteSend (new byte [] { 0 }, 1);
                    //await MqttRemoteSend (new byte [] { 0 }, 2);
                    //await MqttRemoteSend (new byte [] { 0 }, 4);
                } catch (Exception ex) {
                    //mes = ex.Message;
                    //options1 异常提示
                    //Console.WriteLine ("Connect error: " + ex.Message);
                    MainPage.LoadingTipShow ("Connect error.");
                    if (IfDEBUG) {
                        MainPage.ShowAlertOnMainThread ("Connect error: " + ex.Message);
                    }
                } finally {
                    //MainPage.ShowAlertOnMainThread ("finally: " + mes);
                    //MainPage.AddTip ("finally: " + mes);
                }
            }
        }
        /// <summary>
        /// 从开始到连接成功,只提示1次
        /// </summary>
        static bool bNeedStartTip = true;
        /// <summary>
        /// 正在获取连接参数...
        /// </summary>
        static void ShowStartTip ()
        {
            if (bNeedStartTip) {
                bNeedStartTip = false;
                if (CommonPage.IsRemote) {
                    //MainPage.AddTip ("Getting remote connection parameters...");
                    MainPage.LoadingTipShow ("Please wait, getting connection parameters...");
                }
            }
        }
        /// <summary>
        /// 
        /// </summary>
        /// <param name="message">附加数据包</param>
        /// <param name="optionType">操作类型:0=网关控制;1=订阅网关数据;2=订阅网关上线数据;3=订阅网关是否在线主题  4=订阅挤下线主题 </param>
        /// <param name="optionType">操作类型:0=网关控制;1=订阅网关数据;2=订阅网关上线数据</param>
        /// <returns></returns>
        public static async Task MqttRemoteSend (byte [] message, int optionType = 0)
        {
            try {
                if (!remoteIsConnected) {
                    System.Console.WriteLine ($"============>Mqtt 未连接 取消发送");
                    return;
                }
                //if (!remoteIsConnected) {
                //    Utlis.WriteLine ($"============>Mqtt 未连接 取消发送");
                //    return;
                //}
                var topicName = @"/" + MainPage.LoginUser.AccountString.ToLower () + @"/" + UserConfig.Instance.GatewayMAC.Replace (".", "") + @"/" + currentGuid;
                string topicName;
                switch (optionType) {
                case 0:
                    var messageSend = message;
                    if (string.IsNullOrEmpty (mqttEncryptKey)) {
                        topicName = $"/ClientToBusGateWay/{CurRemoteMACInfo.macMark}/Common/OldON";
                    } else {
                        topicName = $"/ClientToBusGateWay/{CurRemoteMACInfo.macMark}/Common/NewON";
                        messageSend = Shared.Securitys.EncryptionService.AesEncryptPayload (message, mqttEncryptKey);
                    topicName = $"/ClientToBusGateWay/{CurRemoteMACInfo.macMark}/Common/ON";
                    if (!string.IsNullOrEmpty (mqttEncryptKey)) {
                        message = Shared.Securitys.EncryptionService.AesEncryptPayload (message, mqttEncryptKey);
                    }
                    //base64加密
                    var m = new MqttApplicationMessage { Topic = topicName, Payload = messageSend, Retain = false, QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce };
                    //if (remoteIsConnected) {
                    try {
                         RemoteMqttClient.PublishAsync (m);
                    } catch (Exception e) {
                        //await DisConnectRemoteMqttClient (e.Message);
                        //await StartCloudMqtt ();
                        //if (remoteIsConnected) {
                        //    RemoteMqttClient.PublishAsync (m);
                        //}
                    }
                    //}
                    await RemoteMqttClient.PublishAsync (new MqttApplicationMessage { Topic = topicName, Payload = message, Retain = false, QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.AtMostOnce });
                    break;
                case 3:
                    topicName = $"/ClientToBusGateWay/{CurRemoteMACInfo.macMark}/Common/CheckGateway";
                    var m1 = new MqttApplicationMessage { Topic = topicName, Retain = false, QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.ExactlyOnce };
                    try {
                        Console.WriteLine ("CheckGateway");
                         RemoteMqttClient.PublishAsync (m1);
                    } catch (Exception e) {
                        Console.WriteLine ($"CheckGateway Fail:{e.Message}");
                        //await DisConnectRemoteMqttClient (e.Message);
                        //await StartCloudMqtt ();
                    }
                    //Utlis.WriteLine ("CheckGateway");
                    await RemoteMqttClient.PublishAsync (new MqttApplicationMessage { Topic = topicName, Retain = false, QualityOfServiceLevel = MQTTnet.Protocol.MqttQualityOfServiceLevel.AtMostOnce });
                    break;
                }
            } catch (Exception e) {
                // Utlis.WriteLine ($"============>Mqtt MqttRemoteSend catch");
                 if (!IsDisConnectingWithSendCatch) {
                    IsDisConnectingWithSendCatch = true;
                    await DisConnectRemoteMqttClient ("MqttRemoteSendCatch");
                }
            }
        }
        /// <summary>
        /// SendCatch 后执行一次断开操作
        /// </summary>
        static bool IsDisConnectingWithSendCatch = false;
        /// <summary>
        /// 分享住宅 获取当前住宅网关信息并且连接MQTT 或者刷新  
@@ -472,13 +609,15 @@
                    } else {
                        System.Console.WriteLine ("============>Mqtt 取消连接,当前住宅没绑定网关");
                        Utlis.WriteLine ("============>Mqtt 取消连接,当前住宅没绑定网关");
                    }
                } else {
                    var mGatewayRes = infoResult.PageData.Find ((obj) => obj.GatewayUniqueId == UserConfig.Instance.CurrentRegion.HomeGateways [0].GatewayUniqueId);
                    if (mGatewayRes != null) {
                        CurRemoteMACInfo.aesKey = mGatewayRes.AesKey;
                        mqttEncryptKey = CurRemoteMACInfo.aesKey;
                    } else {
                        MainPage.LoadingTipShow ("Please wait, failed to get gateway parameters.");
                    }
                }
@@ -488,41 +627,130 @@
           
        }
        /// <summary>
        /// 是否需要读取一次所有设备状态
        /// </summary>
        static bool IfNeedReadAllDeviceStatus = true;
        /// <summary>
        /// 收到网关上线消息
        /// </summary>
        static async Task ReceiveNotifyBusGateWayInfoChange ()
        {
            //当前住宅不是分享来
            if (!UserConfig.Instance.CurrentRegion.IsOthreShare) {
            Utlis.WriteLine ("============>Mqtt 网关上线");
            SetGatewayOnlineResetCheck ();
            if (CommonPage.IsRemote) {
                var gatewayListUrl = $"{MainPage.RequestHttpsHostMqtt}/Center/Center/GetGatewayPagger"; //App、Buspro软件登录后获取网关列表 http 请求
                var gatewayListRequestPar = new RemoteRequestParameters () { Mac = CurRemoteMACInfo.mac, LoginAccessToken = mqttRequestParToken, RequestVersion = "RequestVersion1", RequestProtocolType = 0, RequestSource = 1 };
                var gatewayListRequestResult = MainPage.RequestHttps ("", Newtonsoft.Json.JsonConvert.SerializeObject (gatewayListRequestPar), gatewayListUrl);
                var gatewayListRequestResult_Obj = Newtonsoft.Json.JsonConvert.DeserializeObject<MqttRemoteInfo> (gatewayListRequestResult.ResponseData.ToString ());
                if (gatewayListRequestResult_Obj != null && gatewayListRequestResult_Obj.pageData.Count > 0) {
                    var mData = gatewayListRequestResult_Obj.pageData.Find ((obj) => obj.mac == UserConfig.Instance.CurrentRegion.HomeGateways [0].GatewayUniqueId);
                    if (mData != null) {
                        CurRemoteMACInfo.aesKey = mData.aesKey;
                        mqttEncryptKey = CurRemoteMACInfo.aesKey;
                    }
//#if DEBUG
                MainPage.AddTip ("Gateway login online");
//#endif
                if (IfNeedReadAllDeviceStatus) {
                    IfNeedReadAllDeviceStatus = false;
                    Shared.SimpleControl.Phone.UserMiddle.ReadAllDeviceStatus ();
                }
            } else {
                GetSingleHomeGatewayPaggerAndMQTTConnectAsync (false);
            }
            //请求刷新一次 网关加密Key
            await GetSingleHomeGatewayPaggerAndMQTTConnectAsync (false);
        }
        /// <summary>
        /// 收到网关掉线信息
        /// </summary>
        static void ReceiveNotifyGateWayOffline ()
        {
            Utlis.WriteLine ("============>Mqtt GateWayOffline");
            IsGatewayOnline = false;
            if (CommonPage.IsRemote) {
                Utlis.ShowAppLinkStatus (AppLinkStatus.CloudOffline);
                MainPage.AddTip ("Remote failed,gateway offline");
                //if (MainPage.WiFiStatus != "CrabtreeAdd/CloudUnlink.png") {
                //    Utlis.ShowAppLinkStatus (AppLinkStatus.CloudUnlink);
                //}
            }
        }
        static int CheckGatewayCount = 0;
        static DateTime mCheckGatewayTime;
        /// <summary>
        /// 设置网关在线标志,并重置CheckGateway参数
        /// </summary>
        static void SetGatewayOnlineResetCheck ()
        {
            IsGatewayOnline = true;
            mCheckGatewayTime = DateTime.Now;
            CheckGatewayCount = 0;
            if (CommonPage.IsRemote) {
                if (MainPage.WiFiStatus != "CrabtreeAdd/CloudLink.png") {
                    Utlis.ShowAppLinkStatus (AppLinkStatus.CloudLink);
                }
            }
        }
        //static void TipGatewayOffline () {
        //}
//        /// <summary>
//        /// 定时检测网关是否在线
//        /// 1.线程休眠间隔5s
//        /// 2.发送CheckGateway间隔为9S,收到主题刷新该时间,重置发送次数
//        /// 3.发送次数到达3次以上,则判定网关为离线状态
//        /// </summary>
//        /// <returns></returns>
//        static void InitCheckGateway ()
//        {
//            new System.Threading.Thread (async () => {
//                while (true) {
//                    try {
//                        if (CommonPage.IsRemote && remoteIsConnected) {
//                            if (CheckGatewayCount >= 3) {
//                                //连续3次没回复,判定网关超时
//                                if (MainPage.WiFiStatus == "CrabtreeAdd/CloudLink.png") {
//                                    //IsGatewayOnline = false;
//                                    Utlis.ShowAppLinkStatus (AppLinkStatus.CloudOffline);
//                                    MainPage.AddTip ("Gateway offline!");
//                                }
//                            }
//                            if (mCheckGatewayTime.AddSeconds (10).Ticks <= System.DateTime.Now.Ticks) {
//                                mCheckGatewayTime = DateTime.Now;
//                                //CheckGateway
//                                MqttRemoteSend (new byte [] { }, 3);
//                                CheckGatewayCount++;
//#if DEBUG
//                                Utlis.WriteLine ("CheckGateway");
//#endif
//                            }
//                        }
//                        System.Threading.Thread.Sleep (5500);
//                    } catch { }
//                }
//            }) { IsBackground = true }.Start ();
//        }
        /// <summary>
        /// 收到CheckGateway主题
        /// </summary>
        static void ReceiveCheckGatewayTopic () {
            //Utlis.WriteLine ("============>Mqtt CheckGateway网关回复");
            SetGatewayOnlineResetCheck ();
        }
        /// <summary>
        /// 收到挤下线推送
        /// </summary>
        static async Task ReceiveNotifySqueezeAsync (byte [] mes)
        {
            Console.WriteLine ($"============>MqttNotifySqueeze: {ErrorCode.LoginInAnotherDevice}");
            Utlis.WriteLine ($"============>MqttNotifySqueeze: {ErrorCode.LoginInAnotherDevice}");
            var mMes = CommonPage.MyEncodingUTF8.GetString (mes);
            if (mMes == PushSignStr) return;//是自己的登录推送不处理
            CommonPage.IsRemote = false;
@@ -540,7 +768,6 @@
            MainPage.ShowAlertOnMainThread (ErrorCode.LoginInAnotherDevice);
            Application.RunOnMainThread (() => {
                MainPage.WiFiStatus = "CrabtreeAdd/WiFi.png";
                UserMiddle.btnLinkStatus.UnSelectedImagePath = MainPage.WiFiStatus;
                new AccountLogin ().Show ();
@@ -553,7 +780,6 @@
public class ShareMemberConnMqttInfoObj : GetConnMqttInfoObj
{
    /// <summary>
    /// 主人用户Id 
    /// </summary>
@@ -580,6 +806,10 @@
    /// 
    /// </summary>
    public string PublishPayloadJsonStr = "";
    /// <summary>
    /// Mac
    /// </summary>
    public string Mac = "";
}
public class RemoteRequestParameters
@@ -613,6 +843,8 @@
    public string connEmqClientId;
    public string connEmqUserName;
    public string connEmqPwd;
    public List<RemoteMACInfo> AccountAllGateways;
}
public class RemoteMACInfo
@@ -625,6 +857,8 @@
    public string groupName;
    public string projectName;
    public string userName;
    public string clientId;
    //app自定义数据
    public string md5_mac_string;
@@ -797,7 +1031,7 @@
                    RemoteMqttClient = null;
                }
                CommonPage.IsRemote = false;
                Console.WriteLine ("Close Mqtt!!!");
                Utlis.WriteLine ("Close Mqtt!!!");
            } catch { }
        }
@@ -887,7 +1121,7 @@
                                        webPush.DeleteToken_Push (UserConfig.Instance.tokenID);
                                    });
                                } catch (Exception ex) {
                                    Console.WriteLine (ex.Message);
                                    Utlis.WriteLine (ex.Message);
                                } finally {
                                    Application.RunOnMainThread (() => {
                                        Shared.SimpleControl.Phone.UserMiddle.LinkStatusTip.BackgroundColor = SkinStyle.Current.LinkStatusTipColor;
@@ -953,7 +1187,7 @@
                        });
                        RemoteMqttClient.UseDisconnectedHandler (e => {
                            //Console.WriteLine ("RemoteMqttClient UseDisconnectedHandler");
                            //Utlis.WriteLine ("RemoteMqttClient UseDisconnectedHandler");
                            if (thisShowTip) {
                                if (CommonPage.IsRemote) {
                                    //MainPage.AddTip (Language.StringByID (Shared.SimpleControl.R.MyInternationalizationString.RemoteConnectionDisconnected));
@@ -967,7 +1201,7 @@
                            }
                        });
                        RemoteMqttClient.UseConnectedHandler (async e => {
                            //Console.WriteLine ("RemoteMqttClient IsRemote");
                            //Utlis.WriteLine ("RemoteMqttClient IsRemote");
                            await MqttCheckGateway ();
                        });
                    }
@@ -980,14 +1214,14 @@
                            MainPage.Loading.Hide ();
                            Shared.SimpleControl.Phone.UserMiddle.LinkStatusTip.BackgroundColor = SkinStyle.Current.DelColor;
                        });
                        System.Console.WriteLine ($"重新连接远程通讯失败,因为获取新的KEY失败");
                        Utlis.WriteLine ($"重新连接远程通讯失败,因为获取新的KEY失败");
                        return;
                    }
                    //var messgae = System.Text.Encoding.UTF8.GetString (tempResult);
                    var responsePack = tempResult.ResponseData;// Newtonsoft.Json.JsonConvert.DeserializeObject<Dictionary<string, object>> (messgae);
                    //if (responsePack == null || responsePack ["ResponseData"] == null) {
                    //    Console.WriteLine ("断开后重新链接需要重新登录获取连接的密码失败");
                    //    Utlis.WriteLine ("断开后重新链接需要重新登录获取连接的密码失败");
                    //    return;
                    //}
                    var dictrionaryResult = Newtonsoft.Json.JsonConvert.DeserializeObject<Dictionary<string, object>> (tempResult.ResponseData.ToString ());
@@ -1015,7 +1249,7 @@
                    await Close ();
                    await RemoteMqttClient.ConnectAsync (options);
                } catch (Exception ex) {
                    System.Console.WriteLine ("============>" + ex.Message);
                    Utlis.WriteLine ("============>" + ex.Message);
                    Application.RunOnMainThread (() => {
                        MainPage.Loading.Hide ();
                        isConnecting = false.ToString ();
@@ -1109,7 +1343,7 @@
                    return;
                }
                var topicName = @"/" + MainPage.LoginUser.AccountString.ToLower () + @"/CheckGateway/" + UserConfig.Instance.GatewayMAC.Replace (".", "");
                Console.WriteLine ("MqttCheckGateway : " + topicName);
                Utlis.WriteLine ("MqttCheckGateway : " + topicName);
                //base64加密主题
                var topicNameBase64 = Shared.Securitys.EncryptionService.AesEncryptTopic (topicName, mqttEncryptKey);