MQTT 服务错误 "socket failed: EPERM"

MQTT Service error "socket failed: EPERM"

这是我的 class MQTTCONECTION

 public class MQTTService extends Service {
        private static final String TAG = "MQTTService";
        private static boolean hasWifi = false;
        private static boolean hasMmobile = false;
        private Thread thread;
        private ConnectivityManager mConnMan;
        private volatile IMqttAsyncClient mqttClient;
        private String deviceId;

    class MQTTBroadcastReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(Context context, Intent intent) {
            IMqttToken token;
            boolean hasConnectivity = false;
            boolean hasChanged = false;
            NetworkInfo infos[] = mConnMan.getAllNetworkInfo();

            for (int i = 0; i < infos.length; i++){
                if (infos[i].getTypeName().equalsIgnoreCase("MOBILE")){
                    if((infos[i].isConnected() != hasMmobile)){
                        hasChanged = true;
                        hasMmobile = infos[i].isConnected();
                    }
                    Log.d(TAG, infos[i].getTypeName() + " is " + infos[i].isConnected());
                } else if ( infos[i].getTypeName().equalsIgnoreCase("WIFI") ){
                    if((infos[i].isConnected() != hasWifi)){
                        hasChanged = true;
                        hasWifi = infos[i].isConnected();
                    }
                    Log.d(TAG, infos[i].getTypeName() + " is " + infos[i].isConnected());
                }
            }

            hasConnectivity = hasMmobile || hasWifi;
            Log.v(TAG, "hasConn: " + hasConnectivity + " hasChange: " + hasChanged + " -                  "+(mqttClient == null || !mqttClient.isConnected()));
            if (hasConnectivity && hasChanged && (mqttClient == null || !mqttClient.isConnected())) {
                //Log.d(TAG, "Llama a la función doConnect");
                doConnect();


            } else if (!hasConnectivity && mqttClient != null && mqttClient.isConnected()) {
                Log.d(TAG, "doDisconnect()");
                try {
                    token = mqttClient.disconnect();
                    token.waitForCompletion(1000);
                } catch (MqttException e) {
                    e.printStackTrace();
                }
            }
        }
    };

    public class MQTTBinder extends Binder {
        public MQTTService getService(){
            return MQTTService.this;
        }
    }

    @Override
    public void onCreate() {
        IntentFilter intentf = new IntentFilter();
        setClientID();
        intentf.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
        registerReceiver(new MQTTBroadcastReceiver(), new IntentFilter(ConnectivityManager.CONNECTIVITY_ACTION));
        mConnMan = (ConnectivityManager) getSystemService(CONNECTIVITY_SERVICE);
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        Log.d(TAG, "onConfigurationChanged()");
        android.os.Debug.waitForDebugger();
        super.onConfigurationChanged(newConfig);

    }

    private void setClientID(){
        WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
        WifiInfo wInfo = wifiManager.getConnectionInfo();
        deviceId = wInfo.getMacAddress();
        if(deviceId == null){
            deviceId = MqttAsyncClient.generateClientId();
        }
    }

    private void doConnect(){

        IMqttToken token;
        MqttConnectOptions options = new MqttConnectOptions();
        options.setCleanSession(true);
        //StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().detectDiskReads().detectDiskWrites().detectNetwork().penaltyLog().build());

        try {
            mqttClient = new MqttAsyncClient("tcp://192.168.0.9:1883", deviceId, new MemoryPersistence());
            token = mqttClient.connect();
            token.waitForCompletion(3500);
            mqttClient.setCallback(new MqttEventCallback());
            token = mqttClient.subscribe("test", 0);
            token.waitForCompletion(5000);
            token.setActionCallback(new IMqttActionListener() {

                @Override
                public void onSuccess(IMqttToken asyncActionToken) {
                    Log.d(TAG, "Si se conecto");
                    /*String mensaje = "Conchita";
                    byte[] mensajeB = mensaje.getBytes(UTF_8);
                    MqttMessage mqttMessage = new MqttMessage(mensajeB);
                    try {
                        mqttClient.publish("test", mqttMessage);
                    } catch (MqttException e) {
                        e.printStackTrace();
                    }*/
                }
                @Override
                public void onFailure(IMqttToken asyncActionToken, Throwable exception) {
                    Log.d(TAG, "Que nel");
                }
            });
        } catch (MqttSecurityException e) {
            e.printStackTrace();
        } catch (MqttException e) {

            switch (e.getReasonCode()) {

                case MqttException.REASON_CODE_BROKER_UNAVAILABLE:
                    Log.d(TAG, "The broker was not available to handle the request.");
                    break;
                case MqttException.REASON_CODE_CLIENT_ALREADY_DISCONNECTED:
                    Log.d(TAG, "The client is already disconnected.");
                    break;
                case MqttException.REASON_CODE_CLIENT_CLOSED:
                    Log.d(TAG, "The client is closed - no operations are permitted on the client in this state.");
                    break;
                case MqttException.REASON_CODE_CLIENT_DISCONNECT_PROHIBITED:
                    Log.d(TAG, "Thrown when an attempt to call MqttClient.disconnect() has been made from within a method on MqttCallback.");
                    break;
                case MqttException.REASON_CODE_CLIENT_DISCONNECTING:
                    Log.d(TAG, "The client is currently disconnecting and cannot accept any new work.");
                    break;
                case MqttException.REASON_CODE_CLIENT_EXCEPTION:
                    Log.d(TAG, "Client encountered an exception.");
                    System.out.println("Cause of Exception: "
                            + e.getCause());
                    break;
                case MqttException.REASON_CODE_CLIENT_NOT_CONNECTED:
                    Log.d(TAG, "The client is not connected to the server.");
                    break;
                case MqttException.REASON_CODE_CLIENT_TIMEOUT:
                    Log.d(TAG, "Client timed out while waiting for a response from the server.");
                    break;
                case MqttException.REASON_CODE_CONNECT_IN_PROGRESS:
                    Log.d(TAG, "A connect operation in already in progress, only one connect can happen at a time.");
                    break;
                case MqttException.REASON_CODE_CONNECTION_LOST:
                    Log.d(TAG, "The client has been unexpectedly disconnected from the server.");
                    break;
                case MqttException.REASON_CODE_DISCONNECTED_BUFFER_FULL:
                    Log.d(TAG, "The Client has attempted to publish a message whilst in the 'resting' / ");
                    break;
                case MqttException.REASON_CODE_FAILED_AUTHENTICATION:
                    Log.d(TAG, "Client encountered an exception.");
                    break; //
                case MqttException.REASON_CODE_INVALID_CLIENT_ID:
                    Log.d(TAG, "The server has rejected the supplied client ID");
                    break;
                case MqttException.REASON_CODE_INVALID_MESSAGE:
                    Log.d(TAG, "Protocol error: the message was not recognized as a valid MQTT       packet.");
                    break;
                case MqttException.REASON_CODE_INVALID_PROTOCOL_VERSION:
                    Log.d(TAG, "The protocol version requested is not supported by the server.");
                    break;
                case MqttException.REASON_CODE_MAX_INFLIGHT:
                    Log.d(TAG, "A request has been made to send a message but the maximum number of inflight messages has already been reached.");
                    break;
                case MqttException.REASON_CODE_NO_MESSAGE_IDS_AVAILABLE:
                    Log.d(TAG, "Internal error, caused by no new message IDs being available.");
                    break;
                case MqttException.REASON_CODE_NOT_AUTHORIZED:
                    Log.d(TAG, "ot authorized to perform the requested operation");
                    break;//--
                case MqttException.REASON_CODE_SERVER_CONNECT_ERROR:
                    Log.d(TAG, "Unable to connect to server");
                    break;
                case MqttException.REASON_CODE_SOCKET_FACTORY_MISMATCH:
                    Log.d(TAG, "Server URI and supplied SocketFactory do not match.");
                    break;
                case MqttException.REASON_CODE_SSL_CONFIG_ERROR:
                    Log.d(TAG, "SSL configuration error.");
                    break;
                case MqttException.REASON_CODE_SUBSCRIBE_FAILED:
                    Log.d(TAG, "Error from subscribe - returned from the server.");
                    break;
                case MqttException. REASON_CODE_TOKEN_INUSE:
                    Log.d(TAG, "A request has been made to use a token that is already associated with another action.");
                    break;
                case MqttException.REASON_CODE_UNEXPECTED_ERROR:
                    Log.d(TAG, "An unexpected error has occurred.");
                    break;//--
                case MqttException.REASON_CODE_WRITE_TIMEOUT:
                    Log.d(TAG, "Client timed out while waiting to write messages to the server.");
                    break;//--
                /*case MqttException.REASON_CODE_SERVER_CONNECT_ERROR:
                    Log.d(TAG, "c" +e.getMessage());
                    e.printStackTrace();
                    break;
                case MqttException.REASON_CODE_FAILED_AUTHENTICATION:
                    Intent i = new Intent("RAISEALLARM");
                    i.putExtra("ALLARM", e);
                    Log.d(TAG, "b"+ e.getMessage());
                    break;*/
                default:
                    Log.e(TAG, "a_" + e.getMessage());
            }
        }
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.v(TAG, "onStartCommand()");
        return START_STICKY;
    }

    private class MqttEventCallback implements MqttCallback {

        @Override
        public void connectionLost(Throwable arg0) {


        }

        @Override
        public void deliveryComplete(IMqttDeliveryToken arg0) {

        }

        @Override
        @SuppressLint("NewApi")
        public void messageArrived(String topic, final MqttMessage msg) throws Exception {
            Log.i(TAG, "Message arrived from topic" + topic);
            Handler h = new Handler(getMainLooper());
            h.post(new Runnable() {
                @Override
                public void run() {
                    Intent launchA = new Intent(MQTTService.this, MainActivity.class);
                    launchA.putExtra("message", msg.getPayload());
                    //TODO write somethinkg that has some sense
                    if(Build.VERSION.SDK_INT >= 11){
                        launchA.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_REORDER_TO_FRONT|Intent.FLAG_ACTIVITY_NO_ANIMATION);
                    } /*else {
                        launchA.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                    }*/
                    startActivity(launchA);
                    Toast.makeText(getApplicationContext(), "MQTT Message:\n" + new String(msg.getPayload()), Toast.LENGTH_SHORT).show();
                }
            });
        }
    }

    public String getThread(){
        return Long.valueOf(thread.getId()).toString();
    }

    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG, "onBind called");
        return null;
    }

    }

这是我的 MainActivity:

    public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Intent mymqttservice_intent = new Intent(this, MQTTService.class);
        startService(mymqttservice_intent);
    }
    }

和清单文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.serviciomqtt_2">

    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
    <uses-permission android:name="android.permission.ACCESS_INTERNET"/>

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:usesCleartextTraffic="true"
        android:theme="@style/AppTheme">

        <service
            android:enabled="true"
            android:name="MQTTService"
            />


        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

我正在使用 API 29,此外,MOsquitto mqtt 代理处于活动状态

Status Mosquitto Broker

当我 运行 应用程序出现下一个错误时:

I/System.out:异常原因:java.net.SocketException:套接字失败:EPERM(不允许操作)

替换:

<uses-permission android:name="android.permission.ACCESS_INTERNET"/>

与:

<uses-permission android:name="android.permission.INTERNET"/>