【问题标题】:ApprtcDemo with local server works between browsers but not Android native to browserApprtcDemo 与本地服务器在浏览器之间工作,但不是浏览器原生的 Android
【发布时间】:2014-01-13 06:37:02
【问题描述】:

我正在开发一个聊天应用程序并完成它。现在我也想实现视频聊天。 经过大量研究后,我决定使用“WebRTC”库。

我做了什么?

1) 能够在本地服务器上运行 AppRtcDemo 并且在浏览器之间运行良好。

参考:http://www.webrtc.org/reference/getting-started

2)能够构建Android AppRtcDemo。但是当我运行它时说“跨源不支持”。

经过研究,我在 webrtc 讨论中发现,要解决此问题,我需要设置自己的转向服务器。

3) 所以我安装了 webrtc 推荐的最新 rfc5766TurnServer。我成功运行了turn server。

参考:http://code.google.com/p/rfc5766-turn-server/

我对 ApprtcDemo(Web)和(Android)进行了以下更改以与我的 Turn 服务器一起使用

1) apprtc.py

替换:

turn_url = 'https://computeengineondemand.appspot.com/'
turn_url = turn_url + 'turn?' + 'username=' + user + '&key=4080218913'

指向我的轮到服务器:

turn_url = 'http://192.168.5.85:3478/?service=turn&username=biraj'

2) index.html

替换:

var pcConfig = {{ pc_config|safe }};

与:

var pcConfig = {"iceServers": [{"url": "stun:stun.l.google.com:19302"},            {"url":"turn:biraj@192.168.5.85:3479", "credential":"0x5b04123c3eec4cf0be64ab909bb2ff5b"}]};

安卓

1)AppRTCDemoActivity.java

替换:

roomInput.setText("https://apprtc.appspot.com/?r=");

使用我的本地 apprtc 服务器:

roomInput.setText("http://192.168.5.86:8080/?r=");

2) AppRTCClient.java

private PeerConnection.IceServer requestTurnServer(String url){}函数中

替换:

connection.addRequestProperty("origin", "https://apprtc.appspot.com");

与:

connection.addRequestProperty("origin", "http://192.168.5.86:8080");

3) /assets/channel.html

替换:

<script src="https://apprtc.appspot.com/_ah/channel/jsapi"></script>

与:

<script src="http://192.168.5.86:8080/_ah/channel/jsapi"></script>

现在我的问题是为什么这在浏览器之间有效,但在 android AppRtcDemo 和浏览器之间无效。

当我在进行上述更改后在 android 上运行 AppRtcDemo 时,本地摄像头预览在右上角开始并且消息提示“等待 ICEcandidates”然后什么也没有发生。

提前致谢。

感谢所有人支持我的问题。在 ApprtcDemo 经历了漫长的艰难旅程后,我取得了成功,并且运行良好。我正在发布解决方案。

找到“GAEChannelClient.java”java文件。

并进行如下更改。

/*
 * libjingle
 * Copyright 2013, Google Inc.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *  1. Redistributions of source code must retain the above copyright notice,
 *     this list of conditions and the following disclaimer.
 *  2. Redistributions in binary form must reproduce the above copyright notice,
 *     this list of conditions and the following disclaimer in the documentation
 *     and/or other materials provided with the distribution.
 *  3. The name of the author may not be used to endorse or promote products
 *     derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package org.appspot.apprtc;

import java.io.InputStream;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.util.Log;
import android.webkit.ConsoleMessage;
import android.webkit.JavascriptInterface;
import android.webkit.WebChromeClient;
import android.webkit.WebView;
import android.webkit.WebViewClient;

/**
 * Java-land version of Google AppEngine's JavaScript Channel API:
 * https://developers.google.com/appengine/docs/python/channel/javascript
 * 
 * Requires a hosted HTML page that opens the desired channel and dispatches JS
 * on{Open,Message,Close,Error}() events to a global object named
 * "androidMessageHandler".
 */
public class GAEChannelClient {
    private static final String TAG = "GAEChannelClient";
    private WebView webView;
    private final ProxyingMessageHandler proxyingMessageHandler;

    /**
     * Callback interface for messages delivered on the Google AppEngine
     * channel.
     * 
     * Methods are guaranteed to be invoked on the UI thread of |activity|
     * passed to GAEChannelClient's constructor.
     */
    public interface MessageHandler {
        public void onOpen();

        public void onMessage(String data);

        public void onClose();

        public void onError(int code, String description);
    }

    /** Asynchronously open an AppEngine channel. */
    @SuppressLint("SetJavaScriptEnabled")
    public GAEChannelClient(Activity activity, String token, MessageHandler handler) {
        webView = new WebView(activity);

        webView.getSettings().setJavaScriptEnabled(true);
        webView.getSettings().setAllowFileAccessFromFileURLs(true); // Maybe you
                                                                    // don't
                                                                    // need this
                                                                    // rule
        webView.getSettings().setAllowUniversalAccessFromFileURLs(true);

        webView.setWebChromeClient(new WebChromeClient() { // Purely for
                                                            // debugging.
            public boolean onConsoleMessage(ConsoleMessage msg) {
                Log.d(TAG, "console: " + msg.message() + " at " + msg.sourceId() + ":" + msg.lineNumber());
                return false;
            }
        });
        webView.setWebViewClient(new WebViewClient() { // Purely for debugging.
            public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
                Log.e(TAG, "JS error: " + errorCode + " in " + failingUrl + ", desc: " + description);
            }

            @Override
            public boolean shouldOverrideUrlLoading(WebView view, String url) {
                System.out.println("HI");
                return super.shouldOverrideUrlLoading(view, url);
            }
        });

        proxyingMessageHandler = new ProxyingMessageHandler(activity, handler, token);
        webView.addJavascriptInterface(proxyingMessageHandler, "androidMessageHandler");
//       webView.loadUrl("file:///android_asset/channel.html");
        try {
            InputStream is = activity.getAssets().open("channel.html");
            StringBuilder builder = new StringBuilder();
            byte[] buffer = new byte[1024];
            while (is.read(buffer) != -1) {
                builder.append(new String(buffer));
            }
            is.close();
            String str = builder.toString();
            webView.loadDataWithBaseURL("http://192.168.5.86:8080", str, "text/html", "utf-8", null);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    /** Close the connection to the AppEngine channel. */
    public void close() {
        if (webView == null) {
            return;
        }
        proxyingMessageHandler.disconnect();
        webView.removeJavascriptInterface("androidMessageHandler");
        webView.loadUrl("about:blank");
        webView = null;
    }

    // Helper class for proxying callbacks from the Java<->JS interaction
    // (private, background) thread to the Activity's UI thread.
    private static class ProxyingMessageHandler {
        private final Activity activity;
        private final MessageHandler handler;
        private final boolean[] disconnected = { false };
        private final String token;

        public ProxyingMessageHandler(Activity activity, MessageHandler handler, String token) {
            this.activity = activity;
            this.handler = handler;
            this.token = token;
        }

        public void disconnect() {
            disconnected[0] = true;
        }

        private boolean disconnected() {
            return disconnected[0];
        }

        @JavascriptInterface
        public String getToken() {
            return token;
        }

        @JavascriptInterface
        public void onOpen() {

            System.out.println("GAEClient : Open" );
            activity.runOnUiThread(new Runnable() {
                public void run() {
                    if (!disconnected()) {
                        handler.onOpen();
                    }
                }
            });
        }

        @JavascriptInterface
        public void onMessage(final String data) {
            System.out.println("GAEClient : Message : " +data );
            activity.runOnUiThread(new Runnable() {
                public void run() {
                    if (!disconnected()) {
                        handler.onMessage(data);
                    }
                }
            });
        }

        @JavascriptInterface
        public void onClose() {
            System.out.println("GAEClient : Close" );
            activity.runOnUiThread(new Runnable() {
                public void run() {
                    if (!disconnected()) {
                        handler.onClose();
                    }
                }
            });
        }

        @JavascriptInterface
        public void onError(final int code, final String description) {
            System.out.println("GAEClient : Erroe : " + description);
            activity.runOnUiThread(new Runnable() {
                public void run() {
                    if (!disconnected()) {
                        handler.onError(code, description);
                    }
                }
            });
        }
    }
}

Channel.html在资产文件夹中

<html>
  <head>
    <script src="http://192.168.5.86:8080/_ah/channel/jsapi"></script>
  </head>
  <!--
  Helper HTML that redirects Google AppEngine's Channel API to a JS object named
  |androidMessageHandler|, which is expected to be injected into the WebView
  rendering this page by an Android app's class such as AppRTCClient.
  -->
  <body onbeforeunload="closeSocket()" onload="openSocket()">
    <script type="text/javascript">
      var token = androidMessageHandler.getToken();
      if (!token)
        throw "Missing/malformed token parameter: [" + token + "]";

      var channel = null;
      var socket = null;

      function openSocket() {
        channel = new goog.appengine.Channel(token);
        socket = channel.open({
          'onopen': function() { androidMessageHandler.onOpen(); },
          'onmessage': function(msg) { androidMessageHandler.onMessage(msg.data); },
          'onclose': function() { androidMessageHandler.onClose(); },
          'onerror': function(err) { androidMessageHandler.onError(err.code, err.description); }
        });
      }

      function closeSocket() {
        socket.close();
      }
    </script>
  </body>
</html>

【问题讨论】:

  • 在android和web通信过程中你会提供日志吗?
  • @Biraj Zalavadia:就像你花了 1 个月,我已经失去了 3 个月仍然无法使它工作,你提到你的修改对你有用,但我完全像你展示的那样,从未工作过我。您能否也加入我的链接:stackoverflow.com/questions/23949237/…
  • @hushao 问“[H]你是如何创建你的[r]本地服务器的?[它]和[as]apprtc.appspot.com一样吗?”

标签: android webrtc apprtcdemo rfc5766turnserver


【解决方案1】:

很遗憾,我不知道你是否做过这些事情:

  1. 使用 SAME stun 并打开每个应用程序(PC 或移动设备)的服务器。
  2. 您是否在申请之间发送 ICE 候选人(我认为您这样做,但只是为了验证)。
  3. 您确定 STUN/TURN url 是导致错误的原因吗,因为我无法相信这些事情与跨域有关(它们不应该,因为您只是从客户端连接到服务器。跨源主要用于从外部源加载数据的网页上。不允许从 XHR 执行此操作)。我真的认为这与https://apprtc.appspot.com/_ah/channel/jsapi 有关,因为这是一个很好的跨源示例。

如果您在移动设备上的 chrome 浏览器中打开正在运行的网页会怎样?那它有什么作用呢? (请注意,您可以将您的手机连接到您的电脑以使用 chrome 拥有的完整开发者工具。Chrome 在您的 android 设备上运行,但您可以在您的电脑上看到 devtools)。

如果您能提供这些答案,我或许可以帮助您。尝试恢复所有这些更改并仅使用 google 的 TURN 服务器,但仅将 https://apprtc.appspot.com/_ah/channel/jsapi 文件设为本地。

编辑:I see you found your answer。你介意分享一下吗?

【讨论】:

  • 我在 html 文件中添加了 javascript,而不是通过网络加载。但仍然无法正常工作。你能帮忙吗,没有人分享使它起作用的细节。
  • @YumYumYum 究竟是什么不起作用?我已经看到 OP 找到了答案,但我不知道是什么。
  • @MarjinS95:实际上 OP 在这里和其他 url 上报告的内容我做了完全相同的 + 花费超过 1 个月的时间,它不像 OP 所说的那样工作。我的问题出在 AppRTCDemo 中,当我执行连接本地设置时,加入房间后它什么也没做。如果你想我应该打开一个与此相关的新问题并链接你吗?我想你是唯一理解这个问题的人,大多数人都很困惑。
  • @YumYumYum 打开一个新问题会很好(您可以添加更多详细信息,甚至可能记录一些关于执行哪些功能以及不执行哪些功能的日志)。我主要是一个javascripter,没有javaer,所以我对它的了解不多,但大部分都懂。我还想查看一些您正在使用的网址,以了解您是否也有任何跨域问题。
  • 好的 - 我会这样做,但我相信你可以解决它。许多其他专家都忽略了大多数相关问题。但它认为它更多的是与 JavaScript 相关的问题,我将创建一个关于这个的新问题,其中包含我所采取的每一个细节和一步一步。那么今天就在这里通知大家。非常感谢您的支持,我真的很感激。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2016-02-16
  • 1970-01-01
  • 2013-10-14
  • 2014-05-11
  • 1970-01-01
  • 2019-08-26
  • 2017-01-01
相关资源
最近更新 更多