4

私の Android アプリは、ポート 24783 で独自の Web サーバーを実行しています。

ローカルワイヤレスにいるときに接続できます。電話のクロム ブラウザにローカルの wifi の IP とポートを入力すると、ウェブサイトが表示されます。ただし、オフラインの場合 (ただし、サーバーは電話でローカルに実行されています)、127.0.0.1:24783 に接続できません。さらに、4G 経由でのみ接続している場合は、127.0.0.1:24783 に接続できません。すべてのトラフィックがキャリア プロキシを経由し、127.0.0.1 での接続を許可しないためです。

127.0.0.1 ではプロキシと通信しないように電話に伝える方法はありますか? その 127.0.0.1 は常に「ローカル」ですか? 私は自分のウェブビューを喜んで使用します。

アップデート:

これは単なる KitKat の問題であることが判明しました。

私は解決策を見つけていませんが、もう少し近づいてください。

以下の解決策は、アプリを開いたときにのみ機能し、[戻る] ボタンを使用して閉じ、もう一度開きます。

package com.ndream.console;

import android.annotation.TargetApi;
import android.content.Context;
import android.content.Intent;
import android.net.Proxy;
import android.os.Build;
import android.os.Parcelable;
import android.util.ArrayMap;
import android.util.Log;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * Created by andrin on 15/01/15.
 */
public class ProxyUtils {
    @TargetApi(Build.VERSION_CODES.KITKAT)
    public static void setKitKatWebViewProxy(Context appContext, String host, int port) {
        if (Build.VERSION.SDK_INT != 19) {
            return;
        }
        Log.i("Console", "PROXY MANUALLY SET");
        System.setProperty("http.proxyHost", host);
        System.setProperty("http.proxyPort", port + "");
        System.setProperty("https.proxyHost", host);
        System.setProperty("https.proxyPort", port + "");
        try {
            Class applictionCls = Class.forName("android.app.Application");
            Field loadedApkField = applictionCls.getDeclaredField("mLoadedApk");
            loadedApkField.setAccessible(true);
            Object loadedApk = loadedApkField.get(appContext);
            Class loadedApkCls = Class.forName("android.app.LoadedApk");
            Field receiversField = loadedApkCls.getDeclaredField("mReceivers");
            receiversField.setAccessible(true);
            ArrayMap receivers = (ArrayMap) receiversField.get(loadedApk);
            for (Object receiverMap : receivers.values()) {
                for (Object rec : ((ArrayMap) receiverMap).keySet()) {
                    Class clazz = rec.getClass();
                    if (clazz.getName().contains("ProxyChangeListener")) {
                        Method onReceiveMethod = clazz.getDeclaredMethod("onReceive", Context.class, Intent.class);
                        Intent intent = new Intent(Proxy.PROXY_CHANGE_ACTION);

                        /*********** optional, may be need in future *************/
                        final String CLASS_NAME = "android.net.ProxyProperties";
                        Class cls = Class.forName(CLASS_NAME);
                        Constructor constructor = cls.getConstructor(String.class, Integer.TYPE, String.class);
                        constructor.setAccessible(true);
                        Object proxyProperties = constructor.newInstance(host, port, null);
                        intent.putExtra("proxy", (Parcelable) proxyProperties);
                        /*********** optional, may be need in future *************/

                        onReceiveMethod.invoke(rec, appContext, intent);
                    }
                }
            }
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        }
    }
}

そして、起動時に次のように呼び出します。

ProxyUtils.setKitKatWebViewProxy(getApplicationContext(), "", 0);
4

0 に答える 0