概要
WebRTC初学者向けに、SkyWayのチュートリアルを参考にチュートリアルを作ってみました。
SkyWayのチュートリアルに細かい解説もありますので、ぜひ一読ください。
環境
環境はapacheの環境さえあれば問題ないのですが、再現しやすいように今回はDockerを利用しています。
また、見た目を整えるためにBootstrapも利用しています。
- maxOS BigSur 11.2.3
- Docker version 20.10.6
- docker-compose version 1.29.1
- skyway 4.4.1
- Bootstrap 4.3.1
構成
最終的には以下のような構成になります。
SkyWayやBootstrapはCDNを利用していますので、プロジェクト自体はシンプルです。
.
├── docker-compose.yml
└── web
├── assets
│ └── js
│ └── index.js
└── index.html
成果物
左側に自分の映像、右側に相手の映像が映るようになります。
(画像では、余計なものが映っているので塗りつぶしています。)
HTML
<!DOCTYPE html>
<html lang="ja">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
<!-- Bootstrap CSS -->
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css"
integrity="sha384-ggOyR0iXCbMQv3Xipma34MD+dH/1fQ784/j6cY/iJTQUOhcWr7x9JvoRxT2MZw1T"
crossorigin="anonymous">
<title>WebRTCサンプル</title>
</head>
<body>
<div class="container">
<div class="row">
<h1>WebRTCサンプル</h1>
</div>
<div class="row">
<div class="col-6">
<div class="form-group">
<p class="col-form-label">自分のPeerID: <span id="my-id"></span></p>
</div>
<video id="my-video" width="100%" autoplay muted playsinline></video>
</div>
<div class="col-6">
<div class="form-group row">
<label class="col-3 col-form-label" for="their-id">相手のPeerID: </label>
<div class="col-7">
<input id="their-id" class="form-control">
</div>
<div class="col-2">
<button type="button" class="btn btn-primary" id="make-call">発信</button>
</div>
</div>
<video id="their-video" width="100%" autoplay muted playsinline></video>
</div>
</div>
</div>
<!-- Optional JavaScript -->
<script src="https://cdn.webrtc.ecl.ntt.com/skyway-4.4.1.js"></script>
<script src="assets/js/index.js"></script>
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
<script src="https://code.jquery.com/jquery-3.3.1.slim.min.js"
integrity="sha384-q8i/X+965DzO0rT7abK41JStQIAqVgRVzpbzo5smXKp4YfRvH+8abtTE1Pi6jizo"
crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.14.7/umd/popper.min.js"
integrity="sha384-UO2eT0CpHqdSJQ6hJty5KVphtPhzWj9WO1clHTMGa3JDZwrnQq4sF86dIHNDz0W1"
crossorigin="anonymous"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"
integrity="sha384-JjSmVgyd0p3pXB1rRibZUAYoIIy6OrQ6VrjIEaFf/nJGzIxFDsf4x0xIM+B07jRM"
crossorigin="anonymous"></script>
</body>
</html>
JS
let localStream;
// カメラ映像取得
navigator.mediaDevices.getUserMedia({video: true, audio: true})
.then(stream => {
// 成功時にvideo要素にカメラ映像をセットし、再生
const videoElm = document.getElementById('my-video');
videoElm.srcObject = stream;
videoElm.play();
// 着信時に相手にカメラ映像を返せるように、グローバル変数に保存しておく
localStream = stream;
})
.catch(error => {
// 失敗時にはエラーログを出力
console.error('mediaDevice.getUserMedia() error:', error);
});
//Peer作成
const peer = new Peer({
key: 'APIキー',
debug: 3
});
// 自分のPeerID取得
peer.on('open', () => {
document.getElementById('my-id').textContent = peer.id;
});
// 発信処理
document.getElementById('make-call').onclick = () => {
const theirID = document.getElementById('their-id').value;
const mediaConnection = peer.call(theirID, localStream);
setEventListener(mediaConnection);
};
// イベントリスナを設置する関数
const setEventListener = mediaConnection => {
mediaConnection.on('stream', stream => {
// video要素にカメラ映像をセットして再生
const videoElm = document.getElementById('their-video');
videoElm.srcObject = stream;
videoElm.play();
});
}
//着信処理
peer.on('call', mediaConnection => {
mediaConnection.answer(localStream);
setEventListener(mediaConnection);
});
作成手順
1. 準備
1-1. SkyWayアカウントの作成、APIキー取得、利用可能ドメイン追加
SkyWayにてアカウントを作成しましょう。
アカウントを作成してログインしたら、「アプリケーション一覧」より「新しくアプリケーションを追加する」をクリックします。
登録画面では、「アプリケーション説明文」と「利用可能ドメイン名」を以下のように入力します。
権限は今回はデフォルトのままで、「アプリケーションを作成する」をクリックしましょう。
すると以下のようにAPIキーが発行されますのでコピーしておきましょう。
1-2. 環境構築
今回はHTMLやJSが動けばいいので、apache用のコンテナのみを用意します。
まずは、docker-compose.ymlを作ります。
version: "3.9"
services:
web:
image: httpd
volumes:
- ./web:/usr/local/apache2/htdocs/
ports:
- "80:80"
そして、確認用のindex.htmlをドキュメントルートに作ります。
BootstrapのCSSとJSをそれぞれ読み込むようにしています。
<!DOCTYPE html>
<html lang="ja">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, shrink-to-fit=no">
<!-- Bootstrap CSS -->
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css"
integrity="sha384-ggOyR0iXCbMQv3Xipma34MD+dH/1fQ784/j6cY/iJTQUOhcWr7x9JvoRxT2MZw1T"
crossorigin="anonymous">
<title>WebRTCサンプル</title>
</head>
<body>
<div class="container">
<div class="row">
<h1>WebRTCサンプル</h1>
</div>
</div>
<!-- Optional JavaScript -->
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
<script src="https://code.jquery.com/jquery-3.3.1.slim.min.js"
integrity="sha384-q8i/X+965DzO0rT7abK41JStQIAqVgRVzpbzo5smXKp4YfRvH+8abtTE1Pi6jizo"
crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.14.7/umd/popper.min.js"
integrity="sha384-UO2eT0CpHqdSJQ6hJty5KVphtPhzWj9WO1clHTMGa3JDZwrnQq4sF86dIHNDz0W1"
crossorigin="anonymous"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"
integrity="sha384-JjSmVgyd0p3pXB1rRibZUAYoIIy6OrQ6VrjIEaFf/nJGzIxFDsf4x0xIM+B07jRM"
crossorigin="anonymous"></script>
</body>
</html>
dockerを起動してみましょう。
docker-compose.ymlがあるディレクトリにて以下のコマンドを実行してください。
docker-compose up -d
ちなみに、dockerを停止させる場合は同じディレクトリにて以下のコマンドを実行してください。
docker-compose down
http://localhost にアクセスしてみましょう。
「WebRTCサンプル」と表示されていれば環境構築はOKです。
2. カメラ映像、マイク音声の取得
2-1. HTML作成
index.html
にvideoタグとscriptタグを追加します。
<div class="container">
<div class="row">
<h1>WebRTCサンプル</h1>
</div>
<div class="row">
<div class="col-6">
<video id="my-video" width="100%" autoplay muted playsinline></video>
</div>
</div>
</div>
<!-- Optional JavaScript -->
<script src="assets/js/index.js"></script>
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
ハウリングを防ぐために、video要素にはmuted属性を加えています。
2-2. JS作成
続いては、index.js
をassets/js
に作ります。
let localStream;
// カメラ映像取得
navigator.mediaDevices.getUserMedia({video: true, audio: true})
.then(stream => {
// 成功時にvideo要素にカメラ映像をセットし、再生
const videoElm = document.getElementById('my-video');
videoElm.srcObject = stream;
videoElm.play();
// 着信時に相手にカメラ映像を返せるように、グローバル変数に保存しておく
localStream = stream;
}).catch(error => {
// 失敗時にはエラーログを出力
console.error('mediaDevice.getUserMedia() error:', error);
});
では、 http://localhost にアクセスしてみましょう。
すると、以下のようにカメラを使用して良いか確認が出ます。
許可
あなたの顔が映っていればOKです。(笑)
ブロック
以下のメッセージがコンソールに出力されます。
index.js:7 mediaDevices.getUserMedia() error: DOMException: Permission denied
3. Peerの作成
3-1. SDKのインポートとAPIキーの設定
相手との通信を行うための処理を追加していきます。
また、SkyWayのSDKをCDNでインポートします。
SDKをインポートします。
SkyWayのCDNはindex.js
より前に記述してください。
<!-- Optional JavaScript -->
<script src="https://cdn.webrtc.ecl.ntt.com/skyway-4.4.1.js"></script>
<script src="assets/js/index.js"></script>
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
SkyWayで作成したアプリケーションのAPIキーを設定してください。
navigator.mediaDevices.getUserMedia({video: true, audio: true})
...
});
//Peer作成
const peer = new Peer({
key: 'APIキー',
debug: 3
});
3-2. PeerID取得
相手と接続するためのPeerIDを取得します。
index.htmlに出力用のタグを用意します。
<div class="col-6">
<div class="form-group">
<p class="col-form-label">自分のPeerID: <span id="my-id"></span></p>
</div>
<video id="my-video" width="100%" autoplay muted playsinline></video>
</div>
//Peer作成
const peer = new Peer({
key: 'APIキー',
debug: 3
});
//PeerID取得
peer.on('open', () => {
document.getElementById('my-id').textContent = peer.id;
});
http://localhost にアクセスしてみてください。
画面上に16桁のランダム英数字が表示されれば成功です。
3-3. 発信・着信処理
HTMLに以下の要素を追加します。
- 相手のPeerIDを記入するテキストエリア
- 発信ボタン
- 着信時に相手の映像を表示するvideo要素
<div class="row">
<div class="col-6">
<div class="form-group">
<p class="col-form-label">自分のPeerID: <span id="my-id"></span></p>
</div>
<video id="my-video" width="100%" autoplay muted playsinline></video>
</div>
<div class="col-6">
<div class="form-group row">
<label class="col-3 col-form-label" for="their-id">相手のPeerID: </label>
<div class="col-7">
<input id="their-id" class="form-control">
</div>
<div class="col-2">
<button type="button" class="btn btn-primary" id="make-call">発信</button>
</div>
</div>
<video id="their-video" width="100%" autoplay muted playsinline></video>
</div>
</div>
// 発信処理
document.getElementById('make-call').onclick = () => {
const theirID = document.getElementById('their-id').value;
const mediaConnection = peer.call(theirID, localStream);
setEventListener(mediaConnection);
};
// イベントリスナを設置する関数
const setEventListener = mediaConnection => {
mediaConnection.on('stream', stream => {
// video要素にカメラ映像をセットして再生
const videoElm = document.getElementById('their-video');
videoElm.srcObject = stream;
videoElm.play();
});
}
//着信処理
peer.on('call', mediaConnection => {
mediaConnection.answer(localStream);
setEventListener(mediaConnection);
});
4. 確認
別タブでページを2つ表示して、他方のPeerIDをテキストエリアにコピー&ペーストし、発信ボタンを押下しましょう。
どちらのページにも映像が2つ表示されれば疎通成功です。
これで完成です。