video-auth-js-sdk
Version:
A SDK to authenticate users with camera through a realtime stream
455 lines (409 loc) • 22 kB
JavaScript
'use strict';
import App from "./App";
import {CallTopicManager} from "./call/callTopicManager";
import buildConfig from "./buildConfig.json"
import {errorList} from "./errorHandler";
import Utility from "./utility/utility"
class VideoAuthStream {
constructor({
token,
async: {
appId,
deviceId,
socketAddress,
messageTtl = 0,
serverName
},
// faceApi: faceApiConfig = {},
enableFaceBox = true,
// onSuccess = null,
onError = null,
onDebug = null,
onReady = null,
videoContainerId
}) {
// const {
// modelPath = 'assets/model/'
// } = faceApiConfig;
this._app = new App({
socketAddress,
appId,
serverName,
deviceId,
token,
// onSuccess,
onError,
onDebug,
onReady,
messageTtl: messageTtl || 0,
videoContainerId,
// modelPath,
enableFaceBox
});
this._isDestroyed = false;
this._faceImage = null;
// this._app.events.on(this._app.events.eventsList.AI_MESSAGE, this._processAiMessage.bind(this));
this._app.events.on(this._app.events.eventsList.CALL_MESSAGE, this._processCallMessage.bind(this));
this._app.events.on(this._app.events.eventsList.ASYNC_READY, this._requestAuthSession.bind(this));
// this.errorList = errorList;
}
_processAiMessage(message){
this._app.publicCallbacks.onDebug({
type: 'PROGRESS',
message: `Received message from ai-simulator`,
data: message
});
switch(message.actionId) {
case 1:
this._app.authSessionInfo.callId = message.callId;
this._app.authSessionInfo.clientId = message.clientId;
this._app.authSessionInfo.brokerAddress = message.brokerAddress;
this._app.authSessionInfo.brokerAddressWeb = message.brokerAddressWeb;
this._app.authSessionInfo.turnAddress = message.turnAddress;
this._app.authSessionInfo.generateTurnsList(message.turnAddress);
this._app.authSessionInfo.topicSend = message.topicSend;
this._app.authSessionInfo.kurentoAddress = message.kurentoAddress;
this._createSessionInCallServer();
break;
case 2:
if (message.callId == this._app.authSessionInfo.callId) {
if(message.done == true) {
this._app.publicCallbacks.onSuccess({
sessionId: this._app.authSessionInfo.callId,
done: true
});
} else {
this._app.publicCallbacks.onError({
sessionId: this._app.authSessionInfo.callId,
done: false,
message: message.reason,
code: errorList.SERVER_PROGRESS_ERROR.code
});
}
this.destroy();
}
break;
}
}
_requestAuthSession(){
this._app.publicCallbacks.onDebug({
type: 'PROGRESS',
message: `Async is ready.`,
});
this._app.publicCallbacks.onReady();
let parentNode = document.getElementById(this._app.params.videoContainerId);
if(parentNode) {
let internalParentNode = document.createElement('div');
internalParentNode.classList.add('auth-video-container');
internalParentNode.style.position = 'relative';
internalParentNode.style.width = '100%';
internalParentNode.style.height = '100%';
this._app.store.videoParentTag = internalParentNode;
// this._app.store.videoParentTag = parentNode;
this._app.store.videoTag = document.createElement('video');
this._app.store.videoTag.classList.add('auth');
this._app.store.videoTag.classList.add('video');
this._app.store.videoTag.id = "video-auth-video-tag";
this._app.store.videoTag.style.width = '100%';
this._app.store.videoTag.style.height = '100%';
internalParentNode.appendChild(this._app.store.videoTag)
// this._app.store.canvasTag = document.createElement('canvas');
// this._app.store.canvasTag.classList.add('auth');
// this._app.store.canvasTag.classList.add('canvas');
// this._app.store.canvasTag.id = "video-auth-canvas-tag";
// this._app.store.canvasTag.style.position = "absolute";
// this._app.store.canvasTag.style.top = "0";
// this._app.store.canvasTag.style.left = "0";
if(this._app.params.enableFaceBox) {
this._faceImage = document.createElement('img');
this._faceImage.src = '';
this._faceImage.classList.add('face-box');
this._faceImage.id = "video-auth-img-tag";
this._faceImage.style.position = "absolute";
this._faceImage.style.top = "0";
this._faceImage.style.left = "0";
internalParentNode.appendChild(this._faceImage);
}
// internalParentNode.appendChild(this._app.store.canvasTag);
parentNode.appendChild(internalParentNode);
this._app.store.externalParentTag = parentNode;
// setTimeout(() => {
// this._app.store.videoTag.srcObject = this._app.store.localCameraStream;
// this._app.store.videoTag.play();
this._setupCamera();
// }, 1000);
} else {
console.error("[SDK] Error: can not find the video parent tag.")
}
}
async _setupCamera() {
const video = this._app.store.videoTag;
if (!video ) return null;
// setup webcam. note that navigator.mediaDevices requires that page is accessed via https
if (!navigator.mediaDevices) {
console.error('Camera Error: access not supported');
this._app.publicCallbacks.onError(errorList.MEDIA_DEVICES_NOT_SUPPORTED);
return null;
}
const constraints = { audio: false, video: { facingMode: 'user', exact: "environment", resizeMode: 'crop-and-scale' } };
if (window.innerWidth > window.innerHeight)
constraints.video.width = {ideal: window.innerWidth }; //width: 1920, height: 1280 }//ideal: 1280}//window.innerWidth };
else
constraints.video.height = { ideal: window.innerHeight };
try {
this._app.store.localCameraStream = await navigator.mediaDevices.getUserMedia(constraints);
} catch (err) {
if (err.name === 'PermissionDeniedError' || err.name === 'NotAllowedError'){
console.error(`Camera Error: camera permission denied: ${err.message || err}`);
this._app.publicCallbacks.onError(errorList.VIDEO_PERMISSION_ERROR);
}
if (err.name === 'SourceUnavailableError') {
console.error(`Camera Error: camera not available: ${err.message || err}`);
this._app.publicCallbacks.onError(this._app.errorHandler.getFilledErrorObject({
...errorList.CAMERA_NOT_AVAILABLE,
replacements: [(err.message || err)]
}));
}
return null;
}
if (this._app.store.localCameraStream) {
video.srcObject = this._app.store.localCameraStream;
} else {
console.error('Camera Error: stream empty');
return null;
}
const track = this._app.store.localCameraStream.getVideoTracks()[0];
const settings = track.getSettings();
if (settings.deviceId) delete settings.deviceId;
if (settings.groupId) delete settings.groupId;
// if (settings.aspectRatio) settings.aspectRatio = Math.trunc(100 * settings.aspectRatio) / 100;
console.log(`Camera active: ${track.label}`);
console.log(`Camera settings: `, {settings});
return new Promise((resolve) => {
video.onloadeddata = async () => {
if(this._app.params.enableFaceBox) {
this._calculateImageSize();
}
setTimeout(()=>{
video.play();
resolve(true);
});
};
});
}
_calculateImageSize(){
let base = "videoWidth";
const video = this._app.store.videoTag;
if(video.videoWidth < video.videoHeight) {
base = "videoHeight";
}
// let aspect = video.videoWidth / video.videoHeight;
if(base == "videoWidth") {
// this._faceImage.width = video.offsetWidth;
// this._faceImage.height = parseInt(video.offsetWidth / aspect);
this._faceImage.style.left = parseInt((video.offsetWidth - this._faceImage.width) / 2) + 'px';
this._faceImage.style.top = parseInt((video.offsetHeight - this._faceImage.height) / 2) + 'px';
} else {
// this._faceImage.height = video.offsetHeight;
// this._faceImage.width = parseInt(video.offsetHeight / aspect);
this._faceImage.style.top = parseInt((video.offsetHeight - this._faceImage.height) / 2) + 'px';
this._faceImage.style.left = parseInt((video.offsetWidth - this._faceImage.width) / 2) + 'px';
}
}
_createSessionInCallServer() {
let that = this,
message = {
id: 'CREATE_SESSION',
brokerAddress: this._app.authSessionInfo.brokerAddressWeb,
turnAddress: this._app.authSessionInfo.turnAddress,
chatId: this._app.authSessionInfo.callId,
};
this._app.messenger.sendCallMessage(message);
}
_processCallMessage(message) {
let uniqueId = message.uniqueId;
if (message.done !== 'FALSE' || (message.done === 'FALSE' && message.desc === 'duplicated')) {
// this._app.store.asyncRequestTimeouts[uniqueId] && clearTimeout(this._app.store.asyncRequestTimeouts[uniqueId]);
} else if (message.done === 'FALSE') {
console.error({
type: 'CALL_ERROR',
code: 7000,
message: "Call Server error: " + (message.desc ? message.desc : message.message),
});
}
switch (message.id) {
case 'PROCESS_SDP_ANSWER':
this.handleProcessSdpAnswer(message);
break;
case 'ADD_ICE_CANDIDATE':
this.handleAddIceCandidate(message);
break;
case 'GET_KEY_FRAME':
// user.videoTopicManager().restartMediaOnKeyFrame(app.store.user.get().id, [2000, 4000, 8000, 12000]);
break;
case 'STOP':
if (this._app.store.messagesCallbacks[uniqueId]) {
this._app.store.messagesCallbacks[uniqueId](message);
}
break;
case 'CLOSE':
if (this._app.store.messagesCallbacks[uniqueId]) {
this._app.store.messagesCallbacks[uniqueId](message);
}
break;
case 'SESSION_NEW_CREATED':
case 'SESSION_REFRESH':
this._handleCreateSessionResult(message);
break;
case 'RECEIVEMETADATA':
// handleReceivedMetaData(message, uniqueId);
break;
case 'ERROR':
// publicized.raiseCallError(app.errorHandler.getFilledErrorObject({
// ...errorList.CALL_SERVER_ERROR,
// replacements: [JSON.stringify(message)]
// }), null, true);
break;
case 'SEND_SDP_OFFER':
case 'RECIVE_SDP_OFFER':
case 'SDP_ANSWER_RECEIVED':
break;
default:
console.warn("[SDK][onmessage] Invalid message, id: " + message.id, message);
// if (jsonMessage.match(/NOT CREATE SESSION/g)) {
// if (currentCallParams && Object.keys(currentCallParams)) {
// //handleCallSocketOpen(currentCallParams);
// callStateController.createSessionInChat(currentCallParams);
// }
// }
break;
}
}
_handleCreateSessionResult(res){
if (res.done === 'TRUE') {
this._topicManager = new CallTopicManager({
app: this._app,
callId: this._app.authSessionInfo.callId,
topic: 'Vi-' + this._app.authSessionInfo.topicSend,
mediaType: 'video',
direction: 'send',
onPeerConnect(){}
});
this._topicManager.createTopic();
}
}
handleProcessSdpAnswer(jsonMessage) {
let peer = this._topicManager.getPeer();
if (peer == null) {
console.error({
type: 'CALL_ERROR',
code: 7000,
message: "[handleProcessSdpAnswer] Skip, no WebRTC Peer",
});
return;
}
peer.processAnswer(jsonMessage.sdpAnswer, (err) => {
if (err) {
console.error({
type: 'CALL_ERROR',
message: "[handleProcessSdpAnswer] Error: " + err,
});
this._app.publicCallbacks.onError({
code: errorList.FAILED_TO_OPEN_PEER.code,
message: "[handleProcessSdpAnswer] Error: " + err
});
return;
}
});
}
handleAddIceCandidate(jsonMessage) {
let peer = this._topicManager.getPeer();
if (!peer) {
console.error({
type: 'CALL_ERROR',
message: "[handleAddIceCandidate] Skip, no WebRTC Peer",
error: JSON.stringify(peer)
});
return;
}
peer.addIceCandidate(jsonMessage.candidate, (err) => {
if (err) {
console.error("[handleAddIceCandidate] " + err);
console.error({
type: 'CALL_ERROR',
message: "[handleAddIceCandidate] " + err,
error: JSON.stringify(jsonMessage.candidate),
});
return;
}
});
}
isFaceDetected() {
return this._app.store.faceDetected;
}
startServerAuthentication(callConfig) {
this._app.authSessionInfo.callId = callConfig.callId;
this._app.authSessionInfo.clientId = callConfig.clientId;
this._app.authSessionInfo.brokerAddress = callConfig.brokerAddress;
this._app.authSessionInfo.brokerAddressWeb = callConfig.brokerAddressWeb;
this._app.authSessionInfo.turnAddress = callConfig.turnAddress;
this._app.authSessionInfo.generateTurnsList(callConfig.turnAddress);
this._app.authSessionInfo.topicSend = callConfig.topicSend;
this._app.authSessionInfo.kurentoAddress = callConfig.kurentoAddress;
this._createSessionInCallServer();
// setTimeout(()=> {
// if(!this._isDestroyed) {
// this._app.publicCallbacks.onSuccess({
// sessionId: this._app.authSessionInfo.callId,
// done: true
// });
// this.destroy();
// }
// }, this._app.authSessionInfo.verificationTime);
// this._app.messenger.sendAiMessage({
// actionId: 1
// });
}
async destroy() {
this._isDestroyed = true;
return new Promise(resolve => {
console.log('innn?')
this._app.messenger.sendCallMessage({
id: 'STOP',
topic: 'Vi-' + this._app.authSessionInfo.topicSend,
clientId: this._app.authSessionInfo.clientId,
});
// this._app.faceApi.destroy();
if(this._app.store.videoParentTag) {
this._app.store.videoParentTag.remove();
this._app.store.videoParentTag = null;
}
if(this._app.store.localCameraStream) {
this._app.store.localCameraStream.getTracks().forEach(item => item.stop());
this._app.store.localCameraStream = null;
}
setTimeout(async function () {
this._topicManager && this._topicManager.destroy();
if (!this._app)
return;
await this._app.async.destroy();
this._app.events.removeAllListeners();
this._app.messenger.clearRequests();
this._app = null;
resolve();
}.bind(this), 1500);
});
}
generateUUID(){
return Utility.generateUUID();
}
version() {
console.log("%c[SDK] Version: video-auth-js-sdk@" + buildConfig.version, "color:green; font-size:13px");
console.log("%c[SDK] Build date:" + buildConfig.date, "color:green;font-size:13px");
console.log("%c[SDK] Additional info: " + buildConfig.VersionInfo, "color:green;font-size:13px");
return buildConfig;
};
}
window.VideoAuthStream = VideoAuthStream;
// export default VideoAuthStream;
module.exports = VideoAuthStream;