aboutsummaryrefslogtreecommitdiffhomepage
path: root/public
diff options
context:
space:
mode:
authorMarvin Borner2019-02-13 17:14:47 +0100
committerMarvin Borner2019-02-13 17:14:47 +0100
commit6a967887df29c0051a22df96463aed63b7ece729 (patch)
tree15b91fccb40990ebc2e0c822b843582e672f0465 /public
parent8cad9c810741088a1874b58e80245ce83909b582 (diff)
Code cleanup
Diffstat (limited to 'public')
-rw-r--r--public/scripts/chat.js505
-rw-r--r--public/scripts/encryption.js490
-rw-r--r--public/scripts/input_pin.js74
-rw-r--r--public/scripts/peer.js2469
-rw-r--r--public/scripts/wordlist.js3273
-rw-r--r--public/styles/pin.sass56
-rw-r--r--public/styles/style.sass56
7 files changed, 3667 insertions, 3256 deletions
diff --git a/public/scripts/chat.js b/public/scripts/chat.js
index 1777318..c1dee3e 100644
--- a/public/scripts/chat.js
+++ b/public/scripts/chat.js
@@ -7,16 +7,19 @@
// general imports
const $ = require('jquery');
+const swal = require('sweetalert');
+const xkcdPassword = require('xkcd-password');
const encryption = require('./encryption');
const wordList = require('./wordlist');
const pinInput = require('./input_pin');
-const swal = require('sweetalert');
-const xkcdPassword = require('xkcd-password');
// setup vars
const host = 'meta.marvinborner.de';
-let peerId, call, fingerprint, connectedPeer;
-let connectedPeers = []; // TODO: Save new peers in array
+let peerId;
+let call;
+let fingerprint;
+let connectedPeer;
+const connectedPeers = []; // TODO: Save new peers in array
// setup generator
const generator = new xkcdPassword();
@@ -26,15 +29,17 @@ generator.initWithWordList(wordList);
* Sets up encryption, user etc.
*/
(async () => {
- // generate peerId
- if (localStorage.getItem('peer_id') === null) {
- peerId = await generator.generate().then(words => words.join('-'));
- localStorage.setItem('peer_id', peerId);
- } else
- peerId = localStorage.getItem('peer_id');
+ // generate peerId
+ if (localStorage.getItem('peer_id') === null) {
+ peerId = await generator.generate()
+ .then(words => words.join('-'));
+ localStorage.setItem('peer_id', peerId);
+ } else {
+ peerId = localStorage.getItem('peer_id');
+ }
- encryption.setup();
- await evaluateKeyGeneration();
+ encryption.setupDatabase();
+ await evaluateKeyGeneration();
})();
/**
@@ -42,239 +47,305 @@ generator.initWithWordList(wordList);
* @returns {Promise<void>}
*/
async function evaluateKeyGeneration() {
- if (localStorage.getItem('database') === 'success' && await encryption.check()) {
- pinInput.init(async (pin, tryCount) => {
- try {
- if (await encryption.getId(await encryption.getPublic()) !== peerId ||
- await encryption.getFingerprint(await encryption.getPublic()) !== await encryption.generateFingerprint(pin)) throw "Not verified!";
- fingerprint = await encryption.generateFingerprint(pin);
- await encryption.decryptPrivate(await encryption.getPrivate(), fingerprint);
- chat()
- } catch (err) { // decrypting failed
- if (tryCount === 3) {
- encryption.reset();
- console.error('Too many tries!');
- pinInput.failure('This account got removed, the site will reload.');
- setTimeout(() => location.reload(true), 1500)
- } else if (err === 'Not verified!') {
- console.error(err);
- pinInput.failure(err);
- } else {
- console.error(err);
- pinInput.failure(err.message);
- }
- }
- });
- } else {
- pinInput.init(async pin => {
- console.log('[LOG] No existing keys found! Generating...');
- pinInput.generate();
- fingerprint = await encryption.generateFingerprint(pin);
- await encryption.generate(peerId, fingerprint).then(() => chat())
- });
- }
+ if (localStorage.getItem('database') === 'success' && await encryption.isEncrypted()) {
+ pinInput.init(async (pin, tryCount) => {
+ try {
+ if (await encryption.getPublicKeyPeerId(await encryption.getPublicKey()) !== peerId
+ || await encryption.getPublicKeyFingerprint(await encryption.getPublicKey())
+ !== await encryption.generateFingerprint(pin)) {
+ throw 'Not verified!';
+ }
+ fingerprint = await encryption.generateFingerprint(pin);
+ await encryption.decryptPrivateKey(await encryption.getPrivateKey(), fingerprint);
+ chat();
+ } catch (err) { // decrypting failed
+ if (tryCount === 3) {
+ encryption.reset();
+ console.error('Too many tries!');
+ pinInput.failure('This account got removed, the site will reload.');
+ setTimeout(() => location.reload(true), 1500);
+ } else if (err === 'Not verified!') {
+ console.error(err);
+ pinInput.failure(err);
+ } else {
+ console.error(err);
+ pinInput.failure(err.message);
+ }
+ }
+ });
+ } else {
+ pinInput.init(async (pin) => {
+ console.log('[LOG] No existing keys found! Generating...');
+ pinInput.generate();
+ fingerprint = await encryption.generateFingerprint(pin);
+ await encryption.generateKeys(peerId, fingerprint)
+ .then(() => chat());
+ });
+ }
}
/**
* Initializes chat functions
*/
function chat() {
- // hide pin input and display chat
- $('#enter_pin').hide();
- $('#chat').fadeIn();
+ // hide pin input and display chat
+ $('#enter_pin')
+ .hide();
+ $('#chat')
+ .fadeIn();
+
+ // start the peer
+ const peer = new Peer(peerId, {
+ host,
+ port: 4242,
+ path: '/api',
+ secure: true,
+ debug: 0,
+ });
- // start the peer
- const peer = new Peer(peerId, {host: host, port: 4242, path: '/api', secure: true, debug: 0});
+ // Peer events
+ peer.on('call', call => getMediaStream(stream => call.answer(stream))); // TODO: Ask for call accept
- // Peer events
- peer.on('call', call => getMediaStream(stream => call.answer(stream))); // TODO: Ask for call accept
- peer.on('open', id => {
- console.log('[LOG] Your ID is', id);
- swal('Hello world!', 'Your ID is "' + id + '".\nYou can share this ID with your friends so they can chat with you!', 'success')
+ peer.on('open', (id) => {
+ console.log('[LOG] Your ID is', id);
+ swal(
+ 'Hello world!',
+ `Your ID is "${id}".\nYou can share this ID with your friends so they can chat with you!`,
+ 'success',
+ );
+ });
+
+ peer.on('error', (err) => {
+ console.error(err);
+ if (err.type === 'network') {
+ swal('Connection to server lost!', '', 'error');
+ } else if (err.type === 'browser-incompatible') {
+ swal('Your server is not compatible!', 'Please update or use another browser!', 'error');
+ } else if (err.type === 'peer-unavailable') {
+ swal('Peer could not be found!', '', 'error');
+ } else if (err.type === 'unavailable-id') {
+ swal('Ou snap! Your ID isn\'t available!', '', 'error');
+ } else {
+ swal('Unhandled Error!', `You just threw up this error: ${err.type}`, 'error');
+ }
+ });
+
+ peer.on('connection', async (conn) => {
+ connectedPeer = conn;
+ console.log('[LOG] Connected to', connectedPeer.peer);
+ swal(
+ 'New connection!',
+ `You have successfully connected to the user ${connectedPeer.peer}!`,
+ 'success',
+ );
+ encryption.getMessages(
+ connectedPeer.peer,
+ await encryption.getPublicKeyPeerId(connectedPeer.peer),
+ await encryption.getPrivateKey(), fingerprint,
+ )
+ .then(messages => messages.forEach(async data => await receivedMessage(`${data.message} - ${data.time}`, true)));
+ connectedPeer.on('open', async () => transferKey(await encryption.getPublicKey()));
+ connectedPeer.on('data', async (message) => {
+ console.log('[LOG] Received new message!');
+ await receivedMessage(message);
});
- peer.on('error', err => {
- console.error(err);
- if (err.type === 'network') swal('Connection to server lost!', '', 'error');
- else if (err.type === 'browser-incompatible') swal('Your server is not compatible!', 'Please update or use another browser!', 'error');
- else if (err.type === 'peer-unavailable') swal('Peer could not be found!', '', 'error');
- else if (err.type === 'unavailable-id') swal('Ou snap! Your ID isn\'t available!', '', 'error');
- else swal('Unhandled Error!', 'You just threw up this error: ' + err.type, 'error');
+ });
+
+ /**
+ * Connects to a peer via his id
+ * @param id
+ * @returns {Promise<void>}
+ */
+ async function connect(id) {
+ const connectionId = (await generator.generate()).join('-');
+ console.log('[LOG] Connecting to', id);
+ console.log('[LOG] Your connection ID is', connectionId);
+ connectedPeer = peer.connect(id, { label: connectionId });
+ console.log('[LOG] Connected with', connectedPeer.peer);
+ encryption.getMessages(
+ connectedPeer.peer,
+ await encryption.getPublicKeyPeerId(connectedPeer.peer),
+ await encryption.getPrivateKey(), fingerprint,
+ )
+ .then(messages => messages.forEach(async data => await receivedMessage(`${data.message} - ${data.time}`, true)));
+ connectedPeer.on('open', async () => {
+ swal(`Successfully connected to "${connectedPeer.peer}"!`, '', 'success');
+ transferKey(await encryption.getPublicKey());
});
- peer.on('connection', async conn => {
- connectedPeer = conn;
- console.log('[LOG] Connected to', connectedPeer.peer);
- swal('New connection!', `You have successfully connected to the user ${connectedPeer.peer}!`, 'success');
- encryption.getMsgs(connectedPeer.peer, await encryption.get(connectedPeer.peer), await encryption.getPrivate(), fingerprint).then(messages =>
- messages.forEach(async data => await receivedMessage(`${data.message} - ${data.time}`, true))
- );
- connectedPeer.on('open', async () => transferKey(await encryption.getPublic()));
- connectedPeer.on('data', async message => {
- console.log('[LOG] Received new message!');
- await receivedMessage(message);
- })
+ connectedPeer.on('data', async (message) => {
+ console.log('[LOG] Received new message!');
+ await receivedMessage(message);
});
+ }
- /**
- * Connects to a peer via his id
- * @param id
- * @returns {Promise<void>}
- */
- async function connect(id) {
- const connectionId = (await generator.generate()).join('-');
- console.log('[LOG] Connecting to', id);
- console.log('[LOG] Your connection ID is', connectionId);
- connectedPeer = peer.connect(id, {label: connectionId});
- console.log('[LOG] Connected with', connectedPeer.peer);
- encryption.getMsgs(connectedPeer.peer, await encryption.get(connectedPeer.peer), await encryption.getPrivate(), fingerprint).then(messages =>
- messages.forEach(async data => await receivedMessage(`${data.message} - ${data.time}`, true))
- );
- connectedPeer.on('open', async () => {
- swal(`Successfully connected to "${connectedPeer.peer}"!`, '', 'success');
- transferKey(await encryption.getPublic())
- });
- connectedPeer.on('data', async message => {
- console.log('[LOG] Received new message!');
- await receivedMessage(message);
- });
- }
-
- /**
- * Sends a message to the peer with which you're currently connected
- * @param message
- * @returns {Promise<void>}
- */
- async function sendMessage(message) {
- console.log(`[LOG] Sending message '${message}' to ${connectedPeer.peer}`);
- connectedPeer.send({
- type: 'text',
- data: await encryption.encrypt(message, await encryption.get(connectedPeer.peer), await encryption.getPrivate(), fingerprint)
- });
- await receivedMessage(message, true);
- }
+ /**
+ * Sends a message to the peer with which you're currently connected
+ * @param message
+ * @returns {Promise<void>}
+ */
+ async function sendMessage(message) {
+ console.log(`[LOG] Sending message '${message}' to ${connectedPeer.peer}`);
+ connectedPeer.send({
+ type: 'text',
+ data: await encryption.encrypt(
+ message,
+ await encryption.getPeerPublicKey(connectedPeer.peer),
+ await encryption.getPrivateKey(), fingerprint,
+ ),
+ });
+ await receivedMessage(message, true);
+ }
- /**
- * Transfers the (public) key to the currently connected peer
- * @param key
- */
- function transferKey(key) {
- console.log(`[LOG] Transferring key to ${connectedPeer.peer}`);
- connectedPeer.send({type: 'key', data: key});
- }
+ /**
+ * Transfers the (public) key to the currently connected peer
+ * @param key
+ */
+ function transferKey(key) {
+ console.log(`[LOG] Transferring key to ${connectedPeer.peer}`);
+ connectedPeer.send({
+ type: 'key',
+ data: key,
+ });
+ }
- /**
- * Renders and processes the incoming messages
- * @param message
- * @param self
- */
- async function receivedMessage(message, self = false) {
- if (self) {
- $('#messages').append(`<span style="color: green">${message}</span><br>`);
- } else {
- if (message.type === 'text') {
- await encryption.storeMsg(connectedPeer.peer, message.data, fingerprint);
- await encryption.decrypt(message.data, await encryption.get(connectedPeer.peer), await encryption.getPrivate(), fingerprint)
- .then(plaintext => $('#messages').append(`<span>${plaintext}</span><br>`));
- } else if (message.type === 'key') {
- await encryption.store(connectedPeer.peer, message.data)
- }
- }
+ /**
+ * Renders and processes the incoming messages
+ * @param message
+ * @param self
+ */
+ async function receivedMessage(message, self = false) {
+ if (self) {
+ $('#messages')
+ .append(`<span style="color: green">${message}</span><br>`);
+ } else if (message.type === 'text') {
+ await encryption.storeMessage(connectedPeer.peer, message.data, fingerprint);
+ await encryption.decrypt(
+ message.data,
+ await encryption.getPeerPublicKey(connectedPeer.peer),
+ await encryption.getPrivateKey(),
+ fingerprint,
+ )
+ .then(plaintext => $('#messages')
+ .append(`<span>${plaintext}</span><br>`));
+ } else if (message.type === 'key') {
+ await encryption.storePeerPublicKey(connectedPeer.peer, message.data);
}
+ }
- /**
- * Sends a message of the text input field
- * @returns {Promise<void>}
- */
- async function sendMessageFromInput() {
- return await sendMessage($('#message').val()) & $('#message').val('')
- }
+ /**
+ * Sends a message of the text input field
+ * @returns {Promise<String>}
+ */
+ async function sendMessageFromInput() {
+ const messageInput = $('#message');
+ return await sendMessage(messageInput.val()) & messageInput.val('');
+ }
- /*+
- * Shows warning modal and deletes account
- */
- function deleteAccount() {
- swal({
- title: 'Are you sure?',
- text: 'Once deleted, you will not be able to recover any messages or connections!',
- icon: 'warning',
- buttons: true,
- dangerMode: true,
- }).then(willDelete => {
- if (willDelete) {
- encryption.reset();
- swal('Successfully deleted your data.', '', 'success').then(() => location.reload(true));
- }
- });
- }
+ /**
+ * Shows warning modal and deletes account
+ */
+ function deleteAccount() {
+ swal({
+ title: 'Are you sure?',
+ text: 'Once deleted, you will not be able to recover any messages or connections!',
+ icon: 'warning',
+ buttons: true,
+ dangerMode: true,
+ })
+ .then((willDelete) => {
+ if (willDelete) {
+ encryption.reset();
+ swal('Successfully deleted your data.', '', 'success')
+ .then(() => location.reload(true));
+ }
+ });
+ }
- /**
- * Shows modal for adding a contact
- */
- function addContact() {
- swal('Add a contact', {
- buttons: true,
- content: 'input',
- attributes: {
- placeholder: 'Contact ID',
- },
- }).then(contactId => {
- if (contactId.match(/^([a-zA-Z]*-[a-zA-Z]*)+$/)) {
- connect(contactId).then(() => swal({
- title: 'Connecting...',
- icon: 'info',
- text: ' ',
- buttons: false,
- closeOnClickOutside: false,
- closeOnEsc: false
- }))
- } else
- swal('Invalid ID!', '', 'error')
- })
- }
+ /**
+ * Shows modal for adding a contact
+ */
+ function addContact() {
+ swal('Add a contact', {
+ buttons: true,
+ content: 'input',
+ attributes: {
+ placeholder: 'Contact ID',
+ },
+ })
+ .then((contactId) => {
+ if (contactId.match(/^([a-zA-Z]*-[a-zA-Z]*)+$/)) {
+ connect(contactId)
+ .then(() => swal({
+ title: 'Connecting...',
+ icon: 'info',
+ text: ' ',
+ buttons: false,
+ closeOnClickOutside: false,
+ closeOnEsc: false,
+ }));
+ } else {
+ swal('Invalid ID!', '', 'error');
+ }
+ });
+ }
- /**
- * Click events
- */
- $(document).ready(() => {
- $('#send_message').on('click', async () => await sendMessageFromInput());
- $('#message').on('keydown', async e => {
- if (e.key === 'Enter') await sendMessageFromInput()
+ /**
+ * Click events
+ */
+ $(document)
+ .ready(() => {
+ $('#send_message')
+ .on('click', async () => await sendMessageFromInput());
+ $('#message')
+ .on('keydown', async (e) => {
+ if (e.key === 'Enter') await sendMessageFromInput();
});
- // FABs
- $('#add_contact').on('click', () => addContact());
- $('#logout').on('click', () => location.reload(true));
- $('#delete').on('click', () => deleteAccount());
- $('#anonymize').on('click', () => {
- if (peer.disconnected) {
- swal('Reconnected to broker server!', 'You can now connect to new peers again.', 'success').then(() =>
- peer.reconnect()
- )
- } else {
- peer.disconnect();
- swal('Disconnected from broker server!', 'You will still be able to send and receive messages.', 'success')
- }
+ // FABs
+ $('#add_contact')
+ .on('click', () => addContact());
+ $('#logout')
+ .on('click', () => location.reload(true));
+ $('#delete')
+ .on('click', () => deleteAccount());
+ $('#anonymize')
+ .on('click', () => {
+ if (peer.disconnected) {
+ swal('Reconnected to broker server!', 'You can now connect to new peers again.', 'success')
+ .then(() => peer.reconnect());
+ } else {
+ peer.disconnect();
+ swal('Disconnected from broker server!', 'You will still be able to send and receive messages.', 'success');
+ }
});
- $('#call').on('click', () => getMediaStream(stream => {
- call = peer.call(peerId, stream); // TODO: Encrypt call
- initCall(call)
+ $('#call')
+ .on('click', () => getMediaStream((stream) => {
+ call = peer.call(peerId, stream); // TODO: Encrypt call
+ initCall(call);
}));
});
}
function getMediaStream(callback) {
- navigator.getUserMedia(
- {audio: true, video: {width: 1280, height: 720}},
- stream => callback(stream),
- err => console.error(err)
- )
+ navigator.getUserMedia(
+ {
+ audio: true,
+ video: {
+ width: 1280,
+ height: 720,
+ },
+ },
+ stream => callback(stream),
+ err => console.error(err),
+ );
}
function initCall(call) {
- call.on('stream', stream => {
- const video = document.querySelector('video');
- video.srcObject = stream;
- video.onloadedmetadata = () => {
- video.play();
- };
- })
+ call.on('stream', (stream) => {
+ const video = document.querySelector('video');
+ video.srcObject = stream;
+ video.onloadedmetadata = () => {
+ video.play();
+ };
+ });
}
diff --git a/public/scripts/encryption.js b/public/scripts/encryption.js
index 377a853..c4bdaed 100644
--- a/public/scripts/encryption.js
+++ b/public/scripts/encryption.js
@@ -8,8 +8,8 @@
const Dexie = require('dexie');
const moment = require('moment');
const crypto = require('crypto');
-const jsSHA = require("jssha");
-const fingerprint = require('fingerprintjs2');
+const JsSHA = require('jssha');
+const fingerprintjs = require('fingerprintjs2');
const openpgp = require('openpgp');
const swal = require('sweetalert');
@@ -18,295 +18,309 @@ let db;
// compress encryption data
openpgp.config.compression = openpgp.enums.compression.zlib;
-/**
- * Generates database and tables
- * @returns Boolean
- */
-function setupDatabase() {
+const self = module.exports = {
+ /**
+ * Generates database and tables
+ * @returns Boolean
+ */
+ setupDatabase: () => {
db = new Dexie('texx');
- db.version(2).stores({
+ db.version(2)
+ .stores({
own_keys: '&key_type, key_data',
peer_keys: 'peer_id, key_data',
- messages: '++id, peer_id, message, time'
- });
+ messages: '++id, peer_id, message, time',
+ });
localStorage.setItem('database', 'success');
- db.open().catch(e => {
+ db.open()
+ .catch((e) => {
localStorage.setItem('database', 'failed');
- console.error("Database failed: " + e.stack);
+ console.error(`Database failed: ${e.stack}`);
swal('Could not create the local database!', 'Please try loading this site from a different browser', 'error');
- });
+ });
return true;
-}
+ },
-/**
- * Generates and stores encrypted private key, public key and a revocation certificate
- * @param peerId
- * @param fingerprint
- * @returns {Promise<void>}
- */
-async function generateKeys(peerId, fingerprint) {
+ /**
+ * Generates and stores encrypted private key, public key and a revocation certificate
+ * @param peerId
+ * @param fingerprint
+ * @returns {Promise<void>}
+ */
+ generateKeys: async (peerId, fingerprint) => {
const options = {
- userIds: [{name: peerId, comment: fingerprint}],
- curve: 'ed25519',
- passphrase: fingerprint
+ userIds: [{
+ name: peerId,
+ comment: fingerprint,
+ }],
+ curve: 'ed25519',
+ passphrase: fingerprint,
};
- openpgp.generateKey(options).then(async (key) => {
- await db.own_keys.put({key_type: 'private_key', key_data: key.privateKeyArmored});
- db.own_keys.put({key_type: 'public_key', key_data: key.publicKeyArmored}).then(() =>
- console.log('[LOG] Successfully generated and stored keys!')
- );
- });
-}
+ openpgp.generateKey(options)
+ .then(async (key) => {
+ await db.own_keys.put({
+ key_type: 'private_key',
+ key_data: key.privateKeyArmored,
+ });
+ db.own_keys.put({
+ key_type: 'public_key',
+ key_data: key.publicKeyArmored,
+ })
+ .then(() => console.log('[LOG] Successfully generated and stored keys!'));
+ });
+ },
-/**
- * /**
- * Gets the peers private key
- * @returns {Dexie.Promise<Dexie.Promise<string>>}
- */
-async function getPrivateKey() {
- return db.own_keys.where('key_type').equals('private_key').limit(1).toArray()
- .then(res => res.length > 0 ? res[0]['key_data'] : '');
-}
+ /**
+ * Gets the peers private key
+ * @returns {Dexie.Promise<Dexie.Promise<string>>}
+ */
+ getPrivateKey: async () => db.own_keys.where('key_type')
+ .equals('private_key')
+ .limit(1)
+ .toArray()
+ .then(res => (res.length > 0 ? res[0].key_data : '')),
-/**
- * Gets the peers public key
- * @returns {Dexie.Promise<Dexie.Promise<string>>}
- */
-async function getPublicKey() {
- return db.own_keys.where('key_type').equals('public_key').limit(1).toArray()
- .then(res => res.length > 0 ? res[0]['key_data'] : '');
-}
+ /**
+ * Gets the peers public key
+ * @returns {Dexie.Promise<Dexie.Promise<string>>}
+ */
+ getPublicKey: async () => db.own_keys.where('key_type')
+ .equals('public_key')
+ .limit(1)
+ .toArray()
+ .then(res => (res.length > 0 ? res[0].key_data : '')),
-/**
- * /**
- * Encrypts the data with a public key (e.g the one of the peer with which you're chatting)
- * @param data
- * @param publicKey
- * @param privateKey
- * @param fingerprint
- * @returns {Promise<String>}
- */
-async function encrypt(data, publicKey, privateKey, fingerprint) {
- const privateKeyObj = await decryptPrivateKey(privateKey, fingerprint);
+ /**
+ * Encrypts the data with a public key (e.g the one of the peer with which you're chatting)
+ * @param data
+ * @param publicKey
+ * @param privateKey
+ * @param fingerprint
+ * @returns {Promise<String>}
+ */
+ encrypt: async (data, publicKey, privateKey, fingerprint) => {
+ const privateKeyObj = await self.decryptPrivateKey(privateKey, fingerprint);
const options = {
- message: openpgp.message.fromText(data),
- publicKeys: (await openpgp.key.readArmored(publicKey)).keys,
- privateKeys: [privateKeyObj] // for signing
+ message: openpgp.message.fromText(data),
+ publicKeys: (await openpgp.key.readArmored(publicKey)).keys,
+ privateKeys: [privateKeyObj], // for signing
};
- return openpgp.encrypt(options).then(ciphertext => ciphertext.data);
-}
+ return openpgp.encrypt(options)
+ .then(ciphertext => ciphertext.data);
+ },
-/**
- * Decrypts encrypted data with own encrypted private key and verifies the data with the public key
- * @param data
- * @param publicKey
- * @param privateKey
- * @param fingerprint
- * @returns {Promise<String>}
- */
-async function decrypt(data, publicKey, privateKey, fingerprint) {
- const privateKeyObj = await decryptPrivateKey(privateKey, fingerprint);
+ /**
+ * Decrypts encrypted data with own encrypted private key and
+ * verifies the data with the public key
+ * @param data
+ * @param publicKey
+ * @param privateKey
+ * @param fingerprint
+ * @returns {Promise<String>}
+ */
+ decrypt: async (data, publicKey, privateKey, fingerprint) => {
+ const privateKeyObj = await self.decryptPrivateKey(privateKey, fingerprint);
const options = {
- message: await openpgp.message.readArmored(data),
- publicKeys: (await openpgp.key.readArmored(publicKey)).keys, // for verification
- privateKeys: [privateKeyObj]
+ message: await openpgp.message.readArmored(data),
+ publicKeys: (await openpgp.key.readArmored(publicKey)).keys, // for verification
+ privateKeys: [privateKeyObj],
};
- return openpgp.decrypt(options).then(plaintext => plaintext.data);
-}
+ return openpgp.decrypt(options)
+ .then(plaintext => plaintext.data);
+ },
-/**
- * Decrypts the private key
- * @param privateKey
- * @param fingerprint
- * @returns {Promise<module:key.Key>}
- */
-async function decryptPrivateKey(privateKey, fingerprint) {
+ /**
+ * Decrypts the private key
+ * @param privateKey
+ * @param fingerprint
+ * @returns {Promise<module:key.Key>}
+ */
+ decryptPrivateKey: async (privateKey, fingerprint) => {
const privateKeyObj = (await openpgp.key.readArmored(privateKey)).keys[0];
await privateKeyObj.decrypt(fingerprint);
return privateKeyObj;
-}
+ },
-/**
- * Checks whether the peer has keys
- * @returns {boolean}
- */
-async function isEncrypted() {
- return Dexie.exists('texx').then(async (exists) => {
- if (exists) {
- const hasPrivateKey = getPrivateKey().then(res => res !== '');
- const hasPublicKey = getPublicKey().then(res => res !== '');
- return (hasPrivateKey && hasPublicKey);
- } else
- return false;
- });
-}
+ /**
+ * Checks whether the peer has keys
+ * @returns {boolean}
+ */
+ isEncrypted: async () => Dexie.exists('texx')
+ .then(async (exists) => {
+ if (exists) {
+ const hasPrivateKey = self.getPrivateKey()
+ .then(res => res !== '');
+ const hasPublicKey = self.getPublicKey()
+ .then(res => res !== '');
+ return (hasPrivateKey && hasPublicKey);
+ }
+ return false;
+ }),
-/**
- * Encrypts a message
- * @param message
- * @param fingerprint
- * @returns {string}
- */
-function encryptMessage(message, fingerprint) {
+ /**
+ * Encrypts a message
+ * @param message
+ * @param fingerprint
+ * @returns {string}
+ */
+ encryptMessage: (message, fingerprint) => {
const cipher = crypto.createCipher('aes-256-ctr', fingerprint);
const plaintext = cipher.update(message, 'utf8', 'hex');
console.log('[LOG] Encrypted message successfully!');
return plaintext + cipher.final('hex');
-}
+ },
-/**
- * Decrypts a message
- * @param message
- * @param fingerprint
- * @returns {string}
- */
-function decryptMessage(message, fingerprint) {
+ /**
+ * Decrypts a message
+ * @param message
+ * @param fingerprint
+ * @returns {string}
+ */
+ decryptMessage: (message, fingerprint) => {
const cipher = crypto.createCipher('aes-256-ctr', fingerprint);
const plaintext = cipher.update(message, 'hex', 'utf8');
console.log('[LOG] Decrypted message successfully!');
return plaintext + cipher.final('utf8');
-}
+ },
-/**
- * Stores a message // TODO: Store and get own messages too
- * @param peerId
- * @param message
- * @param fingerprint
- */
-async function storeMessage(peerId, message, fingerprint) {
- db.messages.put({peer_id: peerId, message: encryptMessage(message, fingerprint), time: new Date()}).then(() =>
- console.log('[LOG] Stored message of ' + peerId)
- );
-}
+ /**
+ * Stores a message // TODO: Store and get own messages too
+ * @param peerId
+ * @param message
+ * @param fingerprint
+ */
+ storeMessage: async (peerId, message, fingerprint) => {
+ db.messages.put({
+ peer_id: peerId,
+ message: self.encryptMessage(message, fingerprint),
+ time: new Date(),
+ })
+ .then(() => console.log(`[LOG] Stored message of ${peerId}`));
+ },
-/**
- * Gets the messages
- * @param peerId
- * @param publicKey
- * @param privateKey
- * @param fingerprint
- * @returns {Promise<Array>}
- */
-async function getMessages(peerId, publicKey, privateKey, fingerprint) {
+ /**
+ * Gets the messages
+ * @param peerId
+ * @param publicKey
+ * @param privateKey
+ * @param fingerprint
+ * @returns {Promise<Array>}
+ */
+ getMessages: async (peerId, publicKey, privateKey, fingerprint) => {
console.log('[LOG] Getting messages...');
try {
- const messages = await db.messages.where('peer_id').equals(peerId).sortBy('id');
- let messageArray = [];
- for (let i = messages.length; i--;) {
- await messageArray.push({
- message: await decrypt(decryptMessage(messages[i]['message'], fingerprint), publicKey, privateKey, fingerprint),
- time: moment(messages[i]['time']).fromNow()
- })
- }
- return messageArray;
+ const messages = await db.messages.where('peer_id')
+ .equals(peerId)
+ .sortBy('id');
+ const messageArray = [];
+ for (let i = messages.length; i--;) {
+ await messageArray.push({
+ message: await self.decrypt(
+ decryptMessage(messages[i].message, fingerprint), publicKey, privateKey, fingerprint,
+ ),
+ time: moment(messages[i].time)
+ .fromNow(),
+ });
+ }
+ return messageArray;
} catch (e) {
- console.log('[LOG] No messages found!');
- return [];
+ console.log('[LOG] No messages found!');
+ return [];
}
-}
+ },
-/**
- * Stores the public key of a peer
- * @param peerId
- * @param key
- */
-async function storePeerPublicKey(peerId, key) {
- db.peer_keys.put({peer_id: peerId, key_data: key}).then(() =>
- console.log('[LOG] Stored public key of ' + peerId)
- );
-}
+ /**
+ * Stores the public key of a peer
+ * @param peerId
+ * @param key
+ */
+ storePeerPublicKey: async (peerId, key) => {
+ db.peer_keys.put({
+ peer_id: peerId,
+ key_data: key,
+ })
+ .then(() => console.log(`[LOG] Stored public key of ${peerId}`));
+ },
-/**
- * Gets and verifies the public key of a peer
- * @param peerId
- * @returns {Dexie.Promise<Dexie.Promise<string>>}
- */
-async function getPeerPublicKey(peerId) {
- return db.peer_keys.where('peer_id').equals(peerId).limit(1).toArray().then(async res => {
- let publicKey;
- if (res.length > 0) {
- publicKey = res[0]['key_data'];
- const publicKeyPeerId = await getPublicKeyPeerId(publicKey);
- if (publicKeyPeerId !== peerId) {
- publicKey = '';
- console.error('[LOG] Public key verification failed! The peers real identity is ' + publicKeyPeerId);
- swal('There\'s something strange going on here!', 'The peers ID could not be verified! His real ID is ' + publicKeyPeerId, 'error');
- } else
- console.log('[LOG] Public key verification succeeded!')
- } else
- publicKey = '';
- return publicKey;
- });
-}
+ /**
+ * Gets and verifies the public key of a peer
+ * @param peerId
+ * @returns {Dexie.Promise<Dexie.Promise<string>>}
+ */
+ getPeerPublicKey: async peerId => db.peer_keys.where('peer_id')
+ .equals(peerId)
+ .limit(1)
+ .toArray()
+ .then(async (res) => {
+ let publicKey;
+ if (res.length > 0) {
+ publicKey = res[0].key_data;
+ const publicKeyPeerId = await self.getPublicKeyPeerId(publicKey);
+ if (publicKeyPeerId !== peerId) {
+ publicKey = '';
+ console.error(`[LOG] Public key verification failed! The peers real identity is ${publicKeyPeerId}`);
+ swal('There\'s something strange going on here!', `The peers ID could not be verified! His real ID is ${publicKeyPeerId}`, 'error');
+ } else {
+ console.log('[LOG] Public key verification succeeded!');
+ }
+ } else {
+ publicKey = '';
+ }
+ return publicKey;
+ }),
-/**
- * Returns peer id of a public key
- * @param publicKey
- * @returns {Promise<String>}
- */
-async function getPublicKeyPeerId(publicKey) {
- return (await openpgp.key.readArmored(publicKey)).keys[0].getPrimaryUser().then(obj => obj.user.userId.userid.replace(/ \((.+?)\)/g, "")) || '';
-}
+ /**
+ * Returns peer id of a public key
+ * @param publicKey
+ * @returns {Promise<String>}
+ */
+ getPublicKeyPeerId: async publicKey => (await openpgp.key.readArmored(publicKey)).keys[0]
+ .getPrimaryUser()
+ .then(obj => obj.user.userId.userid.replace(/ \((.+?)\)/g, '')) || '',
-/**
- * Gets the unique fingerprint of the peer, generated using every data javascript can get from the browser
- * and the hashed passphrase of the peer
- * @param passphrase
- * @returns {Promise<String>}
- */
-function generateFingerprint(passphrase) {
- return fingerprint.getPromise().then(components => {
- const fingerprintHash = fingerprint.x64hash128(components.map(pair => pair.value).join(), 31);
- let shaObj = new jsSHA('SHA3-512', 'TEXT');
- shaObj.update(passphrase);
- const passphraseHash = shaObj.getHash('HEX');
- shaObj = new jsSHA('SHA3-512', 'TEXT');
- shaObj.update(passphraseHash);
- shaObj.update(fingerprintHash);
- return shaObj.getHash('HEX');
- })
-}
+ /**
+ * Gets the unique fingerprint of the peer, generated using every data javascript can get from the
+ * browser and the hashed passphrase of the peer
+ * @param passphrase
+ * @returns {Promise<String>}
+ */
+ generateFingerprint: passphrase => fingerprintjs.getPromise()
+ .then((components) => {
+ const fingerprintHash = fingerprintjs.x64hash128(components.map(pair => pair.value)
+ .join(), 31);
+ let shaObj = new JsSHA('SHA3-512', 'TEXT');
+ shaObj.update(passphrase);
+ const passphraseHash = shaObj.getHash('HEX');
+ shaObj = new JsSHA('SHA3-512', 'TEXT');
+ shaObj.update(passphraseHash);
+ shaObj.update(fingerprintHash);
+ return shaObj.getHash('HEX');
+ }),
-/**
- * Returns fingerprint of a public key
- * @param publicKey
- * @returns {Promise<String>}
- */
-async function getPublicKeyFingerprint(publicKey) {
- return (await openpgp.key.readArmored(publicKey)).keys[0].getPrimaryUser().then(obj => obj.user.userId.userid.match(/\((.*)\)/)[1]) || '';
-}
+ /**
+ * Returns fingerprint of a public key
+ * @param publicKey
+ * @returns {Promise<String>}
+ */
+ getPublicKeyFingerprint: async publicKey => (await openpgp.key.readArmored(publicKey)).keys[0]
+ .getPrimaryUser()
+ .then(obj => obj.user.userId.userid.match(/\((.*)\)/)[1]) || '',
-/**
- * Resets the database/encryption
- */
-function reset() {
+ /**
+ * Resets the database/encryption
+ */
+ reset: () => {
db.delete();
localStorage.removeItem('database');
localStorage.removeItem('peer_id');
- console.log('[LOG] Database has been deleted!')
-}
-
-exports.setup = setupDatabase;
-exports.generate = generateKeys;
-exports.getPrivate = getPrivateKey;
-exports.getPublic = getPublicKey;
-exports.encrypt = encrypt;
-exports.decrypt = decrypt;
-exports.decryptPrivate = decryptPrivateKey;
-exports.check = isEncrypted;
-exports.storeMsg = storeMessage;
-exports.getMsgs = getMessages;
-exports.store = storePeerPublicKey;
-exports.get = getPeerPublicKey;
-exports.getId = getPublicKeyPeerId;
-exports.generateFingerprint = generateFingerprint;
-exports.getFingerprint = getPublicKeyFingerprint;
-exports.reset = reset;
+ console.log('[LOG] Database has been deleted!');
+ },
+};
diff --git a/public/scripts/input_pin.js b/public/scripts/input_pin.js
index c28d6e9..69b068b 100644
--- a/public/scripts/input_pin.js
+++ b/public/scripts/input_pin.js
@@ -6,6 +6,7 @@
*/
const $ = require('jquery');
+
let pin = [];
/**
@@ -13,31 +14,50 @@ let pin = [];
* @param callback
*/
function init(callback) {
- let tryCount = 0;
+ let tryCount = 0;
- $('#1').focus().on('input', (el) => {
- pin.push($(el.target).val());
- $(el.target).val('*');
- $('#' + (parseInt($(el.target).attr('id')) + 1).toString()).focus();
+ $('#1')
+ .focus()
+ .on('input', (el) => {
+ pin.push($(el.target)
+ .val());
+ $(el.target)
+ .val('*');
+ $(`#${(parseInt($(el.target)
+ .attr('id')) + 1).toString()}`)
+ .focus();
});
- $('#2').on('input', (el) => {
- pin.push($(el.target).val());
- $(el.target).val('*');
- $('#' + (parseInt($(el.target).attr('id')) + 1).toString()).focus();
+ $('#2')
+ .on('input', (el) => {
+ pin.push($(el.target)
+ .val());
+ $(el.target)
+ .val('*');
+ $(`#${(parseInt($(el.target)
+ .attr('id')) + 1).toString()}`)
+ .focus();
});
- $('#3').on('input', (el) => {
- pin.push($(el.target).val());
- $(el.target).val('*');
- $('#' + (parseInt($(el.target).attr('id')) + 1).toString()).focus();
+ $('#3')
+ .on('input', (el) => {
+ pin.push($(el.target)
+ .val());
+ $(el.target)
+ .val('*');
+ $(`#${(parseInt($(el.target)
+ .attr('id')) + 1).toString()}`)
+ .focus();
});
- $('#4').on('input', (el) => {
- pin.push($(el.target).val());
- $(el.target).val('*');
- tryCount++;
- callback(pin.join(''), tryCount)
+ $('#4')
+ .on('input', (el) => {
+ pin.push($(el.target)
+ .val());
+ $(el.target)
+ .val('*');
+ tryCount++;
+ callback(pin.join(''), tryCount);
});
}
@@ -45,18 +65,24 @@ function init(callback) {
* Displays error message and clears input
*/
function isWrong(message) {
- $('#pin_message').text(message);
- for (let i = 1; i < 5; i++) $(`#${i}`).val('');
- $('#1').focus();
- pin = [];
+ $('#pin_message')
+ .text(message);
+ for (let i = 1; i < 5; i++) {
+ $(`#${i}`)
+ .val('');
+ }
+ $('#1')
+ .focus();
+ pin = [];
}
/**
* Display generation message
*/
function needsGeneration() {
- // TODO: Add loading animation
- $('#pin_message').text('Generating keys...');
+ // TODO: Add loading animation
+ $('#pin_message')
+ .text('Generating keys...');
}
exports.init = init;
diff --git a/public/scripts/peer.js b/public/scripts/peer.js
index 3b6cd6d..0e03c71 100644
--- a/public/scripts/peer.js
+++ b/public/scripts/peer.js
@@ -1,1121 +1,1414 @@
-/*! peerjs build:0.3.16, production. Copyright(c) 2013 Michelle Bu <michelle@michellebu.com> */
-!function a(b, c, d) {
- function e(g, h) {
- if (!c[g]) {
- if (!b[g]) {
- var i = "function" == typeof require && require;
- if (!h && i) return i(g, !0);
- if (f) return f(g, !0);
- var j = new Error("Cannot find module '" + g + "'");
- throw j.code = "MODULE_NOT_FOUND", j
- }
- var k = c[g] = {exports: {}};
- b[g][0].call(k.exports, function (a) {
- var c = b[g][1][a];
- return e(c || a)
- }, k, k.exports, a, b, c, d)
- }
- return c[g].exports
+/*
+ * peer.js
+ * Copyright (c) 2019, Texx
+ * License: MIT
+ * See https://github.com/texxme/Texx/blob/master/LICENSE
+ */
+!(function a(b, c, d) {
+ function e(g, h) {
+ if (!c[g]) {
+ if (!b[g]) {
+ const i = typeof require === 'function' && require;
+ if (!h && i) return i(g, !0);
+ if (f) return f(g, !0);
+ const j = new Error(`Cannot find module '${g}'`);
+ throw j.code = 'MODULE_NOT_FOUND', j;
+ }
+ const k = c[g] = { exports: {} };
+ b[g][0].call(k.exports, (a) => {
+ const c = b[g][1][a];
+ return e(c || a);
+ }, k, k.exports, a, b, c, d);
}
+ return c[g].exports;
+ }
- for (var f = "function" == typeof require && require, g = 0; g < d.length; g++) e(d[g]);
- return e
+ for (var f = typeof require === 'function' && require, g = 0; g < d.length; g++) e(d[g]);
+ return e;
}({
- 1: [function (a, b, c) {
- b.exports.RTCSessionDescription = window.RTCSessionDescription || window.mozRTCSessionDescription, b.exports.RTCPeerConnection = window.RTCPeerConnection || window.mozRTCPeerConnection || window.webkitRTCPeerConnection, b.exports.RTCIceCandidate = window.RTCIceCandidate || window.mozRTCIceCandidate
- }, {}],
- 2: [function (a, b, c) {
- function d(a, b, c) {
- if (!(this instanceof d)) return new d(a, b, c);
- f.call(this), this.options = e.extend({
- serialization: "binary",
- reliable: !1
- }, c), this.open = !1, this.type = "data", this.peer = a, this.provider = b, this.id = this.options.connectionId || d._idPrefix + e.randomToken(), this.label = this.options.label || this.id, this.metadata = this.options.metadata, this.serialization = this.options.serialization, this.reliable = this.options.reliable, this._buffer = [], this._buffering = !1, this.bufferSize = 0, this._chunkedData = {}, this.options._payload && (this._peerBrowser = this.options._payload.browser), g.startConnection(this, this.options._payload || {originator: !0})
- }
+ 1: [function (a, b, c) {
+ b.exports.RTCSessionDescription = window.RTCSessionDescription || window.mozRTCSessionDescription, b.exports.RTCPeerConnection = window.RTCPeerConnection || window.mozRTCPeerConnection || window.webkitRTCPeerConnection, b.exports.RTCIceCandidate = window.RTCIceCandidate || window.mozRTCIceCandidate;
+ }, {}],
+ 2: [function (a, b, c) {
+ function d(a, b, c) {
+ if (!(this instanceof d)) return new d(a, b, c);
+ f.call(this), this.options = e.extend({
+ serialization: 'binary',
+ reliable: !1,
+ }, c), this.open = !1, this.type = 'data', this.peer = a, this.provider = b, this.id = this.options.connectionId || d._idPrefix + e.randomToken(), this.label = this.options.label || this.id, this.metadata = this.options.metadata, this.serialization = this.options.serialization, this.reliable = this.options.reliable, this._buffer = [], this._buffering = !1, this.bufferSize = 0, this._chunkedData = {}, this.options._payload && (this._peerBrowser = this.options._payload.browser), g.startConnection(this, this.options._payload || { originator: !0 });
+ }
- var e = a("./util"), f = a("eventemitter3"), g = a("./negotiator"), h = a("reliable");
- e.inherits(d, f), d._idPrefix = "dc_", d.prototype.initialize = function (a) {
- this._dc = this.dataChannel = a, this._configureDataChannel()
- }, d.prototype._configureDataChannel = function () {
- var a = this;
- e.supports.sctp && (this._dc.binaryType = "arraybuffer"), this._dc.onopen = function () {
- e.log("Data channel connection success"), a.open = !0, a.emit("open")
- }, !e.supports.sctp && this.reliable && (this._reliable = new h(this._dc, e.debug)), this._reliable ? this._reliable.onmessage = function (b) {
- a.emit("data", b)
- } : this._dc.onmessage = function (b) {
- a._handleDataMessage(b)
- }, this._dc.onclose = function (b) {
- e.log("DataChannel closed for:", a.peer), a.close()
- }
- }, d.prototype._handleDataMessage = function (a) {
- var b = this, c = a.data, d = c.constructor;
- if ("binary" === this.serialization || "binary-utf8" === this.serialization) {
- if (d === Blob) return void e.blobToArrayBuffer(c, function (a) {
- c = e.unpack(a), b.emit("data", c)
- });
- if (d === ArrayBuffer) c = e.unpack(c); else if (d === String) {
- var f = e.binaryStringToArrayBuffer(c);
- c = e.unpack(f)
- }
- } else "json" === this.serialization && (c = JSON.parse(c));
- if (c.__peerData) {
- var g = c.__peerData, h = this._chunkedData[g] || {data: [], count: 0, total: c.total};
- return h.data[c.n] = c.data, h.count += 1, h.total === h.count && (delete this._chunkedData[g], c = new Blob(h.data), this._handleDataMessage({data: c})), void (this._chunkedData[g] = h)
- }
- this.emit("data", c)
- }, d.prototype.close = function () {
- this.open && (this.open = !1, g.cleanup(this), this.emit("close"))
- }, d.prototype.send = function (a, b) {
- if (!this.open) return void this.emit("error", new Error("Connection is not open. You should listen for the `open` event before sending messages."));
- if (this._reliable) return void this._reliable.send(a);
- var c = this;
- if ("json" === this.serialization) this._bufferedSend(JSON.stringify(a)); else if ("binary" === this.serialization || "binary-utf8" === this.serialization) {
- var d = e.pack(a), f = e.chunkedBrowsers[this._peerBrowser] || e.chunkedBrowsers[e.browser];
- if (f && !b && d.size > e.chunkedMTU) return void this._sendChunks(d);
- e.supports.sctp ? e.supports.binaryBlob ? this._bufferedSend(d) : e.blobToArrayBuffer(d, function (a) {
- c._bufferedSend(a)
- }) : e.blobToBinaryString(d, function (a) {
- c._bufferedSend(a)
- })
- } else this._bufferedSend(a)
- }, d.prototype._bufferedSend = function (a) {
- !this._buffering && this._trySend(a) || (this._buffer.push(a), this.bufferSize = this._buffer.length)
- }, d.prototype._trySend = function (a) {
- try {
- this._dc.send(a)
- } catch (a) {
- this._buffering = !0;
- var b = this;
- return setTimeout(function () {
- b._buffering = !1, b._tryBuffer()
- }, 100), !1
- }
- return !0
- }, d.prototype._tryBuffer = function () {
- if (0 !== this._buffer.length) {
- var a = this._buffer[0];
- this._trySend(a) && (this._buffer.shift(), this.bufferSize = this._buffer.length, this._tryBuffer())
- }
- }, d.prototype._sendChunks = function (a) {
- for (var b = e.chunk(a), c = 0, d = b.length; c < d; c += 1) {
- var a = b[c];
- this.send(a, !0)
- }
- }, d.prototype.handleMessage = function (a) {
- var b = a.payload;
- switch (a.type) {
- case"ANSWER":
- this._peerBrowser = b.browser, g.handleSDP(a.type, this, b.sdp);
- break;
- case"CANDIDATE":
- g.handleCandidate(this, b.candidate);
- break;
- default:
- e.warn("Unrecognized message type:", a.type, "from peer:", this.peer)
- }
- }, b.exports = d
- }, {"./negotiator": 5, "./util": 8, eventemitter3: 9, reliable: 12}],
- 3: [function (a, b, c) {
- window.Socket = a("./socket"), window.MediaConnection = a("./mediaconnection"), window.DataConnection = a("./dataconnection"), window.Peer = a("./peer"), window.RTCPeerConnection = a("./adapter").RTCPeerConnection, window.RTCSessionDescription = a("./adapter").RTCSessionDescription, window.RTCIceCandidate = a("./adapter").RTCIceCandidate, window.Negotiator = a("./negotiator"), window.util = a("./util"), window.BinaryPack = a("js-binarypack")
- }, {
- "./adapter": 1,
- "./dataconnection": 2,
- "./mediaconnection": 4,
- "./negotiator": 5,
- "./peer": 6,
- "./socket": 7,
- "./util": 8,
- "js-binarypack": 10
- }],
- 4: [function (a, b, c) {
- function d(a, b, c) {
- if (!(this instanceof d)) return new d(a, b, c);
- f.call(this), this.options = e.extend({}, c), this.open = !1, this.type = "media", this.peer = a, this.provider = b, this.metadata = this.options.metadata, this.localStream = this.options._stream, this.id = this.options.connectionId || d._idPrefix + e.randomToken(), this.localStream && g.startConnection(this, {
- _stream: this.localStream,
- originator: !0
- })
+ var e = a('./util');
+ var f = a('eventemitter3');
+ var g = a('./negotiator');
+ const
+ h = a('reliable');
+ e.inherits(d, f), d._idPrefix = 'dc_', d.prototype.initialize = function (a) {
+ this._dc = this.dataChannel = a, this._configureDataChannel();
+ }, d.prototype._configureDataChannel = function () {
+ const a = this;
+ e.supports.sctp && (this._dc.binaryType = 'arraybuffer'), this._dc.onopen = function () {
+ e.log('Data channel connection success'), a.open = !0, a.emit('open');
+ }, !e.supports.sctp && this.reliable && (this._reliable = new h(this._dc, e.debug)), this._reliable ? this._reliable.onmessage = function (b) {
+ a.emit('data', b);
+ } : this._dc.onmessage = function (b) {
+ a._handleDataMessage(b);
+ }, this._dc.onclose = function (b) {
+ e.log('DataChannel closed for:', a.peer), a.close();
+ };
+ }, d.prototype._handleDataMessage = function (a) {
+ const b = this;
+ let c = a.data;
+ const
+ d = c.constructor;
+ if (this.serialization === 'binary' || this.serialization === 'binary-utf8') {
+ if (d === Blob) {
+ return void e.blobToArrayBuffer(c, (a) => {
+ c = e.unpack(a), b.emit('data', c);
+ });
}
-
- var e = a("./util"), f = a("eventemitter3"), g = a("./negotiator");
- e.inherits(d, f), d._idPrefix = "mc_", d.prototype.addStream = function (a) {
- e.log("Receiving stream", a), this.remoteStream = a, this.emit("stream", a)
- }, d.prototype.handleMessage = function (a) {
- var b = a.payload;
- switch (a.type) {
- case"ANSWER":
- g.handleSDP(a.type, this, b.sdp), this.open = !0;
- break;
- case"CANDIDATE":
- g.handleCandidate(this, b.candidate);
- break;
- default:
- e.warn("Unrecognized message type:", a.type, "from peer:", this.peer)
- }
- }, d.prototype.answer = function (a) {
- if (this.localStream) return void e.warn("Local stream already exists on this MediaConnection. Are you answering a call twice?");
- this.options._payload._stream = a, this.localStream = a, g.startConnection(this, this.options._payload);
- for (var b = this.provider._getMessages(this.id), c = 0, d = b.length; c < d; c += 1) this.handleMessage(b[c]);
- this.open = !0
- }, d.prototype.close = function () {
- this.open && (this.open = !1, g.cleanup(this), this.emit("close"))
- }, b.exports = d
- }, {"./negotiator": 5, "./util": 8, eventemitter3: 9}],
- 5: [function (a, b, c) {
- var d = a("./util"), e = a("./adapter").RTCPeerConnection, f = a("./adapter").RTCSessionDescription,
- g = a("./adapter").RTCIceCandidate, h = {pcs: {data: {}, media: {}}, queue: []};
- h._idPrefix = "pc_", h.startConnection = function (a, b) {
- var c = h._getPeerConnection(a, b);
- if (a.pc = a.peerConnection = c, "media" === a.type && b._stream && c.addStream(b._stream), b.originator) {
- if ("data" === a.type) {
- var e = {};
- d.supports.sctp || (e = {reliable: b.reliable});
- var f = c.createDataChannel(a.label, e);
- a.initialize(f)
- }
- h._makeOffer(a)
- } else h.handleSDP("OFFER", a, b.sdp)
- }, h._getPeerConnection = function (a, b) {
- h.pcs[a.type] || d.error(a.type + " is not a valid connection type. Maybe you overrode the `type` property somewhere."), h.pcs[a.type][a.peer] || (h.pcs[a.type][a.peer] = {});
- var c;
- h.pcs[a.type][a.peer];
- return b.pc && (c = h.pcs[a.type][a.peer][b.pc]), c && "stable" === c.signalingState || (c = h._startPeerConnection(a)), c
- }, h._startPeerConnection = function (a) {
- d.log("Creating RTCPeerConnection.");
- var b = h._idPrefix + d.randomToken(), c = {};
- "data" !== a.type || d.supports.sctp ? "media" === a.type && (c = {optional: [{DtlsSrtpKeyAgreement: !0}]}) : c = {optional: [{RtpDataChannels: !0}]};
- var f = new e(a.provider.options.config, c);
- return h.pcs[a.type][a.peer][b] = f, h._setupListeners(a, f, b), f
- }, h._setupListeners = function (a, b, c) {
- var e = a.peer, f = a.id, g = a.provider;
- d.log("Listening for ICE candidates."), b.onicecandidate = function (b) {
- b.candidate && (d.log("Received ICE candidates for:", a.peer), g.socket.send({
- type: "CANDIDATE",
- payload: {candidate: b.candidate, type: a.type, connectionId: a.id},
- dst: e
- }))
- }, b.oniceconnectionstatechange = function () {
- switch (b.iceConnectionState) {
- case"failed":
- d.log("iceConnectionState is disconnected, closing connections to " + e), a.emit("error", new Error("Negotiation of connection to " + e + " failed.")), a.close();
- break;
- case"disconnected":
- d.log("iceConnectionState is disconnected, closing connections to " + e), a.close();
- break;
- case"completed":
- b.onicecandidate = d.noop
- }
- }, b.onicechange = b.oniceconnectionstatechange, d.log("Listening for data channel"), b.ondatachannel = function (a) {
- d.log("Received data channel");
- var b = a.channel;
- g.getConnection(e, f).initialize(b)
- }, d.log("Listening for remote stream"), b.onaddstream = function (a) {
- d.log("Received remote stream");
- var b = a.stream, c = g.getConnection(e, f);
- "media" === c.type && c.addStream(b)
- }
- }, h.cleanup = function (a) {
- d.log("Cleaning up PeerConnection to " + a.peer);
- var b = a.pc;
- b && (b.readyState && "closed" !== b.readyState || "closed" !== b.signalingState) && (b.close(), a.pc = null)
- }, h._makeOffer = function (a) {
- var b = a.pc;
- b.createOffer(function (c) {
- d.log("Created offer."), !d.supports.sctp && "data" === a.type && a.reliable && (c.sdp = Reliable.higherBandwidthSDP(c.sdp)), b.setLocalDescription(c, function () {
- d.log("Set localDescription: offer", "for:", a.peer), a.provider.socket.send({
- type: "OFFER",
- payload: {
- sdp: c,
- type: a.type,
- label: a.label,
- connectionId: a.id,
- reliable: a.reliable,
- serialization: a.serialization,
- metadata: a.metadata,
- browser: d.browser
- },
- dst: a.peer
- })
- }, function (b) {
- "OperationError: Failed to set local offer sdp: Called in wrong state: kHaveRemoteOffer" != b && (a.provider.emitError("webrtc", b), d.log("Failed to setLocalDescription, ", b))
- })
- }, function (b) {
- a.provider.emitError("webrtc", b), d.log("Failed to createOffer, ", b)
- }, a.options.constraints)
- }, h._makeAnswer = function (a) {
- var b = a.pc;
- b.createAnswer(function (c) {
- d.log("Created answer."), !d.supports.sctp && "data" === a.type && a.reliable && (c.sdp = Reliable.higherBandwidthSDP(c.sdp)), b.setLocalDescription(c, function () {
- d.log("Set localDescription: answer", "for:", a.peer), a.provider.socket.send({
- type: "ANSWER",
- payload: {sdp: c, type: a.type, connectionId: a.id, browser: d.browser},
- dst: a.peer
- })
- }, function (b) {
- a.provider.emitError("webrtc", b), d.log("Failed to setLocalDescription, ", b)
- })
- }, function (b) {
- a.provider.emitError("webrtc", b), d.log("Failed to create answer, ", b)
- })
- }, h.handleSDP = function (a, b, c) {
- c = new f(c);
- var e = b.pc;
- d.log("Setting remote description", c), e.setRemoteDescription(c, function () {
- d.log("Set remoteDescription:", a, "for:", b.peer), "OFFER" === a && h._makeAnswer(b)
- }, function (a) {
- b.provider.emitError("webrtc", a), d.log("Failed to setRemoteDescription, ", a)
- })
- }, h.handleCandidate = function (a, b) {
- var c = b.candidate, e = b.sdpMLineIndex;
- a.pc.addIceCandidate(new g({sdpMLineIndex: e, candidate: c})), d.log("Added ICE candidate for:", a.peer)
- }, b.exports = h
- }, {"./adapter": 1, "./util": 8}],
- 6: [function (a, b, c) {
- function d(a, b) {
- return this instanceof d ? (f.call(this), a && a.constructor == Object ? (b = a, a = void 0) : a && (a = a.toString()), b = e.extend({
- debug: 0,
- host: e.CLOUD_HOST,
- port: e.CLOUD_PORT,
- path: "/",
- token: e.randomToken(),
- config: e.defaultConfig
- }, b), b.key = "peerjs", this.options = b, "/" === b.host && (b.host = window.location.hostname), "/" !== b.path[0] && (b.path = "/" + b.path), "/" !== b.path[b.path.length - 1] && (b.path += "/"), void 0 === b.secure && b.host !== e.CLOUD_HOST && (b.secure = e.isSecure()), b.logFunction && e.setLogFunction(b.logFunction), e.setLogLevel(b.debug), e.supports.audioVideo || e.supports.data ? e.validateId(a) ? (this.destroyed = !1, this.disconnected = !1, this.open = !1, this.connections = {}, this._lostMessages = {}, this._initializeServerConnection(), void (a ? this._initialize(a) : this._retrieveId())) : void this._delayedAbort("invalid-id", 'ID "' + a + '" is invalid') : void this._delayedAbort("browser-incompatible", "The current browser does not support WebRTC")) : new d(a, b)
+ if (d === ArrayBuffer) {
+ c = e.unpack(c);
+ } else if (d === String) {
+ const f = e.binaryStringToArrayBuffer(c);
+ c = e.unpack(f);
}
+ } else {
+ this.serialization === 'json' && (c = JSON.parse(c));
+ }
+ if (c.__peerData) {
+ const g = c.__peerData;
+ const
+ h = this._chunkedData[g] || {
+ data: [],
+ count: 0,
+ total: c.total
+ };
+ return h.data[c.n] = c.data, h.count += 1, h.total === h.count && (delete this._chunkedData[g], c = new Blob(h.data), this._handleDataMessage({ data: c })), void (this._chunkedData[g] = h);
+ }
+ this.emit('data', c);
+ }, d.prototype.close = function () {
+ this.open && (this.open = !1, g.cleanup(this), this.emit('close'));
+ }, d.prototype.send = function (a, b) {
+ if (!this.open) return void this.emit('error', new Error('Connection is not open. You should listen for the `open` event before sending messages.'));
+ if (this._reliable) return void this._reliable.send(a);
+ const c = this;
+ if (this.serialization === 'json') {
+ this._bufferedSend(JSON.stringify(a));
+ } else if (this.serialization === 'binary' || this.serialization === 'binary-utf8') {
+ const d = e.pack(a);
+ const
+ f = e.chunkedBrowsers[this._peerBrowser] || e.chunkedBrowsers[e.browser];
+ if (f && !b && d.size > e.chunkedMTU) return void this._sendChunks(d);
+ e.supports.sctp ? e.supports.binaryBlob ? this._bufferedSend(d) : e.blobToArrayBuffer(d, (a) => {
+ c._bufferedSend(a);
+ }) : e.blobToBinaryString(d, (a) => {
+ c._bufferedSend(a);
+ });
+ } else {
+ this._bufferedSend(a);
+ }
+ }, d.prototype._bufferedSend = function (a) {
+ !this._buffering && this._trySend(a) || (this._buffer.push(a), this.bufferSize = this._buffer.length);
+ }, d.prototype._trySend = function (a) {
+ try {
+ this._dc.send(a);
+ } catch (a) {
+ this._buffering = !0;
+ const b = this;
+ return setTimeout(() => {
+ b._buffering = !1, b._tryBuffer();
+ }, 100), !1;
+ }
+ return !0;
+ }, d.prototype._tryBuffer = function () {
+ if (this._buffer.length !== 0) {
+ const a = this._buffer[0];
+ this._trySend(a) && (this._buffer.shift(), this.bufferSize = this._buffer.length, this._tryBuffer());
+ }
+ }, d.prototype._sendChunks = function (a) {
+ for (let b = e.chunk(a), c = 0, d = b.length; c < d; c += 1) {
+ var a = b[c];
+ this.send(a, !0);
+ }
+ }, d.prototype.handleMessage = function (a) {
+ const b = a.payload;
+ switch (a.type) {
+ case 'ANSWER':
+ this._peerBrowser = b.browser, g.handleSDP(a.type, this, b.sdp);
+ break;
+ case 'CANDIDATE':
+ g.handleCandidate(this, b.candidate);
+ break;
+ default:
+ e.warn('Unrecognized message type:', a.type, 'from peer:', this.peer);
+ }
+ }, b.exports = d;
+ }, {
+ './negotiator': 5,
+ './util': 8,
+ eventemitter3: 9,
+ reliable: 12,
+ }],
+ 3: [function (a, b, c) {
+ window.Socket = a('./socket'), window.MediaConnection = a('./mediaconnection'), window.DataConnection = a('./dataconnection'), window.Peer = a('./peer'), window.RTCPeerConnection = a('./adapter').RTCPeerConnection, window.RTCSessionDescription = a('./adapter').RTCSessionDescription, window.RTCIceCandidate = a('./adapter').RTCIceCandidate, window.Negotiator = a('./negotiator'), window.util = a('./util'), window.BinaryPack = a('js-binarypack');
+ }, {
+ './adapter': 1,
+ './dataconnection': 2,
+ './mediaconnection': 4,
+ './negotiator': 5,
+ './peer': 6,
+ './socket': 7,
+ './util': 8,
+ 'js-binarypack': 10,
+ }],
+ 4: [function (a, b, c) {
+ function d(a, b, c) {
+ if (!(this instanceof d)) return new d(a, b, c);
+ f.call(this), this.options = e.extend({}, c), this.open = !1, this.type = 'media', this.peer = a, this.provider = b, this.metadata = this.options.metadata, this.localStream = this.options._stream, this.id = this.options.connectionId || d._idPrefix + e.randomToken(), this.localStream && g.startConnection(this, {
+ _stream: this.localStream,
+ originator: !0,
+ });
+ }
- var e = a("./util"), f = a("eventemitter3"), g = a("./socket"), h = a("./mediaconnection"),
- i = a("./dataconnection");
- e.inherits(d, f), d.prototype._initializeServerConnection = function () {
- var a = this;
- this.socket = new g(this.options.secure, this.options.host, this.options.port, this.options.path, this.options.key, this.options.wsport), this.socket.on("message", function (b) {
- a._handleMessage(b)
- }), this.socket.on("error", function (b) {
- a._abort("socket-error", b)
- }), this.socket.on("disconnected", function () {
- a.disconnected || (a.emitError("network", "Lost connection to server."), a.disconnect())
- }), this.socket.on("close", function () {
- a.disconnected || a._abort("socket-closed", "Underlying socket is already closed.")
- })
- }, d.prototype._retrieveId = function (a) {
- var b = this, c = new XMLHttpRequest, d = this.options.secure ? "https://" : "http://",
- f = d + this.options.host + ":" + this.options.port + this.options.path + this.options.key + "/id";
- f += "?ts=" + (new Date).getTime() + Math.random(), c.open("get", f, !0), c.onerror = function (a) {
- e.error("Error retrieving ID", a);
- var c = "";
- "/" === b.options.path && b.options.host !== e.CLOUD_HOST && (c = " If you passed in a `path` to your self-hosted PeerServer, you'll also need to pass in that same path when creating a new Peer."), b._abort("server-error", "Could not get an ID from the server." + c)
- }, c.onreadystatechange = function () {
- if (4 === c.readyState) return 200 !== c.status ? void c.onerror() : void b._initialize(c.responseText)
- }, c.send(null)
- }, d.prototype._initialize = function (a) {
- this.id = a, this.socket.start(this.id, this.options.token)
- }, d.prototype._handleMessage = function (a) {
- var b, c = a.type, d = a.payload, f = a.src;
- switch (c) {
- case"OPEN":
- this.emit("open", this.id), this.open = !0;
- break;
- case"ERROR":
- this._abort("server-error", d.msg);
- break;
- case"ID-TAKEN":
- this._abort("unavailable-id", "ID `" + this.id + "` is taken");
- break;
- case"INVALID-KEY":
- this._abort("invalid-key", 'API KEY "' + this.options.key + '" is invalid');
- break;
- case"LEAVE":
- e.log("Received leave message from", f), this._cleanupPeer(f);
- break;
- case"EXPIRE":
- this.emitError("peer-unavailable", "Could not connect to peer " + f);
- break;
- case"OFFER":
- var g = d.connectionId;
- if (b = this.getConnection(f, g), b && (b.close(), e.warn("Offer received for existing Connection ID:", g)), "media" === d.type) b = new h(f, this, {
- connectionId: g,
- _payload: d,
- metadata: d.metadata
- }), this._addConnection(f, b), this.emit("call", b); else {
- if ("data" !== d.type) return void e.warn("Received malformed connection type:", d.type);
- b = new i(f, this, {
- connectionId: g,
- _payload: d,
- metadata: d.metadata,
- label: d.label,
- serialization: d.serialization,
- reliable: d.reliable
- }), this._addConnection(f, b), this.emit("connection", b)
- }
- for (var j = this._getMessages(g), k = 0, l = j.length; k < l; k += 1) b.handleMessage(j[k]);
- break;
- default:
- if (!d) return void e.warn("You received a malformed message from " + f + " of type " + c);
- var m = d.connectionId;
- b = this.getConnection(f, m), b && b.pc ? b.handleMessage(a) : m ? this._storeMessage(m, a) : e.warn("You received an unrecognized message:", a)
- }
- }, d.prototype._storeMessage = function (a, b) {
- this._lostMessages[a] || (this._lostMessages[a] = []), this._lostMessages[a].push(b)
- }, d.prototype._getMessages = function (a) {
- var b = this._lostMessages[a];
- return b ? (delete this._lostMessages[a], b) : []
- }, d.prototype.connect = function (a, b) {
- if (this.disconnected) return e.warn("You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect, or call reconnect on this peer if you believe its ID to still be available."), void this.emitError("disconnected", "Cannot connect to new Peer after disconnecting from server.");
- var c = new i(a, this, b);
- return this._addConnection(a, c), c
- }, d.prototype.call = function (a, b, c) {
- if (this.disconnected) return e.warn("You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect."), void this.emitError("disconnected", "Cannot connect to new Peer after disconnecting from server.");
- if (!b) return void e.error("To call a peer, you must provide a stream from your browser's `getUserMedia`.");
- c = c || {}, c._stream = b;
- var d = new h(a, this, c);
- return this._addConnection(a, d), d
- }, d.prototype._addConnection = function (a, b) {
- this.connections[a] || (this.connections[a] = []), this.connections[a].push(b)
- }, d.prototype.getConnection = function (a, b) {
- var c = this.connections[a];
- if (!c) return null;
- for (var d = 0, e = c.length; d < e; d++) if (c[d].id === b) return c[d];
- return null
- }, d.prototype._delayedAbort = function (a, b) {
- var c = this;
- e.setZeroTimeout(function () {
- c._abort(a, b)
- })
- }, d.prototype._abort = function (a, b) {
- e.error("Aborting!"), this._lastServerId ? this.disconnect() : this.destroy(), this.emitError(a, b)
- }, d.prototype.emitError = function (a, b) {
- e.error("Error:", b), "string" == typeof b && (b = new Error(b)), b.type = a, this.emit("error", b)
- }, d.prototype.destroy = function () {
- this.destroyed || (this._cleanup(), this.disconnect(), this.destroyed = !0)
- }, d.prototype._cleanup = function () {
- if (this.connections) for (var a = Object.keys(this.connections), b = 0, c = a.length; b < c; b++) this._cleanupPeer(a[b]);
- this.emit("close")
- }, d.prototype._cleanupPeer = function (a) {
- for (var b = this.connections[a], c = 0, d = b.length; c < d; c += 1) b[c].close()
- }, d.prototype.disconnect = function () {
- var a = this;
- e.setZeroTimeout(function () {
- a.disconnected || (a.disconnected = !0, a.open = !1, a.socket && a.socket.close(), a.emit("disconnected", a.id), a._lastServerId = a.id, a.id = null)
- })
- }, d.prototype.reconnect = function () {
- if (this.disconnected && !this.destroyed) e.log("Attempting reconnection to server with ID " + this._lastServerId), this.disconnected = !1, this._initializeServerConnection(), this._initialize(this._lastServerId); else {
- if (this.destroyed) throw new Error("This peer cannot reconnect to the server. It has already been destroyed.");
- if (this.disconnected || this.open) throw new Error("Peer " + this.id + " cannot reconnect because it is not disconnected from the server!");
- e.error("In a hurry? We're still trying to make the initial connection!")
- }
- }, d.prototype.listAllPeers = function (a) {
- a = a || function () {
- };
- var b = this, c = new XMLHttpRequest, d = this.options.secure ? "https://" : "http://",
- f = d + this.options.host + ":" + this.options.port + this.options.path + this.options.key + "/peers";
- f += "?ts=" + (new Date).getTime() + Math.random(), c.open("get", f, !0), c.onerror = function (c) {
- b._abort("server-error", "Could not get peers from the server."), a([])
- }, c.onreadystatechange = function () {
- if (4 === c.readyState) {
- if (401 === c.status) {
- var d = "";
- throw d = b.options.host !== e.CLOUD_HOST ? "It looks like you're using the cloud server. You can email team@peerjs.com to enable peer listing for your API key." : "You need to enable `allow_discovery` on your self-hosted PeerServer to use this feature.", a([]), new Error("It doesn't look like you have permission to list peers IDs. " + d)
- }
- a(200 !== c.status ? [] : JSON.parse(c.responseText))
- }
- }, c.send(null)
- }, b.exports = d
- }, {"./dataconnection": 2, "./mediaconnection": 4, "./socket": 7, "./util": 8, eventemitter3: 9}],
- 7: [function (a, b, c) {
- function d(a, b, c, e, g, h) {
- if (!(this instanceof d)) return new d(a, b, c, e, g, h);
- h = h || c, f.call(this), this.disconnected = !1, this._queue = [];
- var i = a ? "https://" : "http://", j = a ? "wss://" : "ws://";
- this._httpUrl = i + b + ":" + c + e + g, this._wsUrl = j + b + ":" + h + e + "peerjs?key=" + g
+ var e = a('./util');
+ var f = a('eventemitter3');
+ var
+ g = a('./negotiator');
+ e.inherits(d, f), d._idPrefix = 'mc_', d.prototype.addStream = function (a) {
+ e.log('Receiving stream', a), this.remoteStream = a, this.emit('stream', a);
+ }, d.prototype.handleMessage = function (a) {
+ const b = a.payload;
+ switch (a.type) {
+ case 'ANSWER':
+ g.handleSDP(a.type, this, b.sdp), this.open = !0;
+ break;
+ case 'CANDIDATE':
+ g.handleCandidate(this, b.candidate);
+ break;
+ default:
+ e.warn('Unrecognized message type:', a.type, 'from peer:', this.peer);
+ }
+ }, d.prototype.answer = function (a) {
+ if (this.localStream) return void e.warn('Local stream already exists on this MediaConnection. Are you answering a call twice?');
+ this.options._payload._stream = a, this.localStream = a, g.startConnection(this, this.options._payload);
+ for (let b = this.provider._getMessages(this.id), c = 0, d = b.length; c < d; c += 1) this.handleMessage(b[c]);
+ this.open = !0;
+ }, d.prototype.close = function () {
+ this.open && (this.open = !1, g.cleanup(this), this.emit('close'));
+ }, b.exports = d;
+ }, {
+ './negotiator': 5,
+ './util': 8,
+ eventemitter3: 9
+ }],
+ 5: [function (a, b, c) {
+ const d = a('./util');
+ const e = a('./adapter').RTCPeerConnection;
+ const f = a('./adapter').RTCSessionDescription;
+
+
+ const g = a('./adapter').RTCIceCandidate;
+ const
+ h = {
+ pcs: {
+ data: {},
+ media: {}
+ },
+ queue: []
+ };
+ h._idPrefix = 'pc_', h.startConnection = function (a, b) {
+ const c = h._getPeerConnection(a, b);
+ if (a.pc = a.peerConnection = c, a.type === 'media' && b._stream && c.addStream(b._stream), b.originator) {
+ if (a.type === 'data') {
+ let e = {};
+ d.supports.sctp || (e = { reliable: b.reliable });
+ const f = c.createDataChannel(a.label, e);
+ a.initialize(f);
}
+ h._makeOffer(a);
+ } else {
+ h.handleSDP('OFFER', a, b.sdp);
+ }
+ }, h._getPeerConnection = function (a, b) {
+ h.pcs[a.type] || d.error(`${a.type} is not a valid connection type. Maybe you overrode the \`type\` property somewhere.`), h.pcs[a.type][a.peer] || (h.pcs[a.type][a.peer] = {});
+ let c;
+ h.pcs[a.type][a.peer];
+ return b.pc && (c = h.pcs[a.type][a.peer][b.pc]), c && c.signalingState === 'stable' || (c = h._startPeerConnection(a)), c;
+ }, h._startPeerConnection = function (a) {
+ d.log('Creating RTCPeerConnection.');
+ const b = h._idPrefix + d.randomToken();
+ let
+ c = {};
+ a.type !== 'data' || d.supports.sctp ? a.type === 'media' && (c = { optional: [{ DtlsSrtpKeyAgreement: !0 }] }) : c = { optional: [{ RtpDataChannels: !0 }] };
+ const f = new e(a.provider.options.config, c);
+ return h.pcs[a.type][a.peer][b] = f, h._setupListeners(a, f, b), f;
+ }, h._setupListeners = function (a, b, c) {
+ const e = a.peer;
+ const f = a.id;
+ const
+ g = a.provider;
+ d.log('Listening for ICE candidates.'), b.onicecandidate = function (b) {
+ b.candidate && (d.log('Received ICE candidates for:', a.peer), g.socket.send({
+ type: 'CANDIDATE',
+ payload: {
+ candidate: b.candidate,
+ type: a.type,
+ connectionId: a.id
+ },
+ dst: e,
+ }));
+ }, b.oniceconnectionstatechange = function () {
+ switch (b.iceConnectionState) {
+ case 'failed':
+ d.log(`iceConnectionState is disconnected, closing connections to ${e}`), a.emit('error', new Error(`Negotiation of connection to ${e} failed.`)), a.close();
+ break;
+ case 'disconnected':
+ d.log(`iceConnectionState is disconnected, closing connections to ${e}`), a.close();
+ break;
+ case 'completed':
+ b.onicecandidate = d.noop;
+ }
+ }, b.onicechange = b.oniceconnectionstatechange, d.log('Listening for data channel'), b.ondatachannel = function (a) {
+ d.log('Received data channel');
+ const b = a.channel;
+ g.getConnection(e, f)
+ .initialize(b);
+ }, d.log('Listening for remote stream'), b.onaddstream = function (a) {
+ d.log('Received remote stream');
+ const b = a.stream;
+ const
+ c = g.getConnection(e, f);
+ c.type === 'media' && c.addStream(b);
+ };
+ }, h.cleanup = function (a) {
+ d.log(`Cleaning up PeerConnection to ${a.peer}`);
+ const b = a.pc;
+ b && (b.readyState && b.readyState !== 'closed' || b.signalingState !== 'closed') && (b.close(), a.pc = null);
+ }, h._makeOffer = function (a) {
+ const b = a.pc;
+ b.createOffer((c) => {
+ d.log('Created offer.'), !d.supports.sctp && a.type === 'data' && a.reliable && (c.sdp = Reliable.higherBandwidthSDP(c.sdp)), b.setLocalDescription(c, () => {
+ d.log('Set localDescription: offer', 'for:', a.peer), a.provider.socket.send({
+ type: 'OFFER',
+ payload: {
+ sdp: c,
+ type: a.type,
+ label: a.label,
+ connectionId: a.id,
+ reliable: a.reliable,
+ serialization: a.serialization,
+ metadata: a.metadata,
+ browser: d.browser,
+ },
+ dst: a.peer,
+ });
+ }, (b) => {
+ b != 'OperationError: Failed to set local offer sdp: Called in wrong state: kHaveRemoteOffer' && (a.provider.emitError('webrtc', b), d.log('Failed to setLocalDescription, ', b));
+ });
+ }, (b) => {
+ a.provider.emitError('webrtc', b), d.log('Failed to createOffer, ', b);
+ }, a.options.constraints);
+ }, h._makeAnswer = function (a) {
+ const b = a.pc;
+ b.createAnswer((c) => {
+ d.log('Created answer.'), !d.supports.sctp && a.type === 'data' && a.reliable && (c.sdp = Reliable.higherBandwidthSDP(c.sdp)), b.setLocalDescription(c, () => {
+ d.log('Set localDescription: answer', 'for:', a.peer), a.provider.socket.send({
+ type: 'ANSWER',
+ payload: {
+ sdp: c,
+ type: a.type,
+ connectionId: a.id,
+ browser: d.browser,
+ },
+ dst: a.peer,
+ });
+ }, (b) => {
+ a.provider.emitError('webrtc', b), d.log('Failed to setLocalDescription, ', b);
+ });
+ }, (b) => {
+ a.provider.emitError('webrtc', b), d.log('Failed to create answer, ', b);
+ });
+ }, h.handleSDP = function (a, b, c) {
+ c = new f(c);
+ const e = b.pc;
+ d.log('Setting remote description', c), e.setRemoteDescription(c, () => {
+ d.log('Set remoteDescription:', a, 'for:', b.peer), a === 'OFFER' && h._makeAnswer(b);
+ }, (a) => {
+ b.provider.emitError('webrtc', a), d.log('Failed to setRemoteDescription, ', a);
+ });
+ }, h.handleCandidate = function (a, b) {
+ const c = b.candidate;
+ const
+ e = b.sdpMLineIndex;
+ a.pc.addIceCandidate(new g({
+ sdpMLineIndex: e,
+ candidate: c
+ })), d.log('Added ICE candidate for:', a.peer);
+ }, b.exports = h;
+ }, {
+ './adapter': 1,
+ './util': 8
+ }],
+ 6: [function (a, b, c) {
+ function d(a, b) {
+ return this instanceof d ? (f.call(this), a && a.constructor == Object ? (b = a, a = void 0) : a && (a = a.toString()), b = e.extend({
+ debug: 0,
+ host: e.CLOUD_HOST,
+ port: e.CLOUD_PORT,
+ path: '/',
+ token: e.randomToken(),
+ config: e.defaultConfig,
+ }, b), b.key = 'peerjs', this.options = b, b.host === '/' && (b.host = window.location.hostname), b.path[0] !== '/' && (b.path = `/${b.path}`), b.path[b.path.length - 1] !== '/' && (b.path += '/'), void 0 === b.secure && b.host !== e.CLOUD_HOST && (b.secure = e.isSecure()), b.logFunction && e.setLogFunction(b.logFunction), e.setLogLevel(b.debug), e.supports.audioVideo || e.supports.data ? e.validateId(a) ? (this.destroyed = !1, this.disconnected = !1, this.open = !1, this.connections = {}, this._lostMessages = {}, this._initializeServerConnection(), void (a ? this._initialize(a) : this._retrieveId())) : void this._delayedAbort('invalid-id', `ID "${a}" is invalid`) : void this._delayedAbort('browser-incompatible', 'The current browser does not support WebRTC')) : new d(a, b);
+ }
- var e = a("./util"), f = a("eventemitter3");
- e.inherits(d, f), d.prototype.start = function (a, b) {
- this.id = a, this._httpUrl += "/" + a + "/" + b, this._wsUrl += "&id=" + a + "&token=" + b, this._startXhrStream(), this._startWebSocket()
- }, d.prototype._startWebSocket = function (a) {
- var b = this;
- this._socket || (this._socket = new WebSocket(this._wsUrl), this._socket.onmessage = function (a) {
- try {
- var c = JSON.parse(a.data)
- } catch (b) {
- return void e.log("Invalid server message", a.data)
- }
- b.emit("message", c)
- }, this._socket.onclose = function (a) {
- e.log("Socket closed."), b.disconnected = !0, b.emit("disconnected")
- }, this._socket.onopen = function () {
- b._timeout && (clearTimeout(b._timeout), setTimeout(function () {
- b._http.abort(), b._http = null
- }, 5e3)), b._sendQueuedMessages(), e.log("Socket open")
- })
- }, d.prototype._startXhrStream = function (a) {
- try {
- var b = this;
- this._http = new XMLHttpRequest, this._http._index = 1, this._http._streamIndex = a || 0, this._http.open("post", this._httpUrl + "/id?i=" + this._http._streamIndex, !0), this._http.onerror = function () {
- clearTimeout(b._timeout), b.emit("disconnected")
- }, this._http.onreadystatechange = function () {
- 2 == this.readyState && this.old ? (this.old.abort(), delete this.old) : this.readyState > 2 && 200 === this.status && this.responseText && b._handleStream(this)
- }, this._http.send(null), this._setHTTPTimeout()
- } catch (a) {
- e.log("XMLHttpRequest not available; defaulting to WebSockets")
- }
- }, d.prototype._handleStream = function (a) {
- var b = a.responseText.split("\n");
- if (a._buffer) for (; a._buffer.length > 0;) {
- var c = a._buffer.shift(), d = b[c];
- try {
- d = JSON.parse(d)
- } catch (b) {
- a._buffer.shift(c);
- break
- }
- this.emit("message", d)
- }
- var f = b[a._index];
- if (f) if (a._index += 1, a._index === b.length) a._buffer || (a._buffer = []), a._buffer.push(a._index - 1); else {
- try {
- f = JSON.parse(f)
- } catch (a) {
- return void e.log("Invalid server message", f)
- }
- this.emit("message", f)
- }
- }, d.prototype._setHTTPTimeout = function () {
- var a = this;
- this._timeout = setTimeout(function () {
- var b = a._http;
- a._wsOpen() ? b.abort() : (a._startXhrStream(b._streamIndex + 1), a._http.old = b)
- }, 25e3)
- }, d.prototype._wsOpen = function () {
- return this._socket && 1 == this._socket.readyState
- }, d.prototype._sendQueuedMessages = function () {
- for (var a = 0, b = this._queue.length; a < b; a += 1) this.send(this._queue[a])
- }, d.prototype.send = function (a) {
- if (!this.disconnected) {
- if (!this.id) return void this._queue.push(a);
- if (!a.type) return void this.emit("error", "Invalid message");
- var b = JSON.stringify(a);
- if (this._wsOpen()) this._socket.send(b); else {
- var c = new XMLHttpRequest, d = this._httpUrl + "/" + a.type.toLowerCase();
- c.open("post", d, !0), c.setRequestHeader("Content-Type", "application/json"), c.send(b)
- }
- }
- }, d.prototype.close = function () {
- !this.disconnected && this._wsOpen() && (this._socket.close(), this.disconnected = !0)
- }, b.exports = d
- }, {"./util": 8, eventemitter3: 9}],
- 8: [function (a, b, c) {
- var d = {iceServers: [{url: "stun:stun.l.google.com:19302"}]}, e = 1, f = a("js-binarypack"),
- g = a("./adapter").RTCPeerConnection, h = {
- noop: function () {
- },
- CLOUD_HOST: "0.peerjs.com",
- CLOUD_PORT: 9e3,
- chunkedBrowsers: {Chrome: 1},
- chunkedMTU: 16300,
- logLevel: 0,
- setLogLevel: function (a) {
- var b = parseInt(a, 10);
- isNaN(parseInt(a, 10)) ? h.logLevel = a ? 3 : 0 : h.logLevel = b, h.log = h.warn = h.error = h.noop, h.logLevel > 0 && (h.error = h._printWith("ERROR")), h.logLevel > 1 && (h.warn = h._printWith("WARNING")), h.logLevel > 2 && (h.log = h._print)
- },
- setLogFunction: function (a) {
- a.constructor !== Function ? h.warn("The log function you passed in is not a function. Defaulting to regular logs.") : h._print = a
- },
- _printWith: function (a) {
- return function () {
- var b = Array.prototype.slice.call(arguments);
- b.unshift(a), h._print.apply(h, b)
- }
- },
- _print: function () {
- var a = !1, b = Array.prototype.slice.call(arguments);
- b.unshift("PeerJS: ");
- for (var c = 0, d = b.length; c < d; c++) b[c] instanceof Error && (b[c] = "(" + b[c].name + ") " + b[c].message, a = !0);
- a ? console.error.apply(console, b) : console.log.apply(console, b)
- },
- defaultConfig: d,
- browser: function () {
- return window.mozRTCPeerConnection ? "Firefox" : window.webkitRTCPeerConnection ? "Chrome" : window.RTCPeerConnection ? "Supported" : "Unsupported"
- }(),
- supports: function () {
- if (void 0 === g) return {};
- var a, b, c = !0, e = !0, f = !1, h = !1, i = !!window.webkitRTCPeerConnection;
- try {
- a = new g(d, {optional: [{RtpDataChannels: !0}]})
- } catch (a) {
- c = !1, e = !1
- }
- if (c) try {
- b = a.createDataChannel("_PEERJSTEST")
- } catch (a) {
- c = !1
- }
- if (c) {
- try {
- b.binaryType = "blob", f = !0
- } catch (a) {
- }
- var j = new g(d, {});
- try {
- h = j.createDataChannel("_PEERJSRELIABLETEST", {}).reliable
- } catch (a) {
- }
- j.close()
- }
- return e && (e = !!a.addStream), a && a.close(), {
- audioVideo: e,
- data: c,
- binaryBlob: f,
- binary: h,
- reliable: h,
- sctp: h,
- onnegotiationneeded: i
- }
- }(),
- validateId: function (a) {
- return !a || /^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(a)
- },
- validateKey: function (a) {
- return !a || /^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(a)
- },
- debug: !1,
- inherits: function (a, b) {
- a.super_ = b, a.prototype = Object.create(b.prototype, {
- constructor: {
- value: a,
- enumerable: !1,
- writable: !0,
- configurable: !0
- }
- })
- },
- extend: function (a, b) {
- for (var c in b) b.hasOwnProperty(c) && (a[c] = b[c]);
- return a
- },
- pack: f.pack,
- unpack: f.unpack,
- log: function () {
- if (h.debug) {
- var a = !1, b = Array.prototype.slice.call(arguments);
- b.unshift("PeerJS: ");
- for (var c = 0, d = b.length; c < d; c++) b[c] instanceof Error && (b[c] = "(" + b[c].name + ") " + b[c].message, a = !0);
- a ? console.error.apply(console, b) : console.log.apply(console, b)
- }
- },
- setZeroTimeout: function (a) {
- function b(b) {
- d.push(b), a.postMessage(e, "*")
- }
+ var e = a('./util');
+ var f = a('eventemitter3');
+ const g = a('./socket');
+ const h = a('./mediaconnection');
- function c(b) {
- b.source == a && b.data == e && (b.stopPropagation && b.stopPropagation(), d.length && d.shift()())
- }
- var d = [], e = "zero-timeout-message";
- return a.addEventListener ? a.addEventListener("message", c, !0) : a.attachEvent && a.attachEvent("onmessage", c), b
- }(window),
- chunk: function (a) {
- for (var b = [], c = a.size, d = index = 0, f = Math.ceil(c / h.chunkedMTU); d < c;) {
- var g = Math.min(c, d + h.chunkedMTU), i = a.slice(d, g),
- j = {__peerData: e, n: index, data: i, total: f};
- b.push(j), d = g, index += 1
- }
- return e += 1, b
- },
- blobToArrayBuffer: function (a, b) {
- var c = new FileReader;
- c.onload = function (a) {
- b(a.target.result)
- }, c.readAsArrayBuffer(a)
- },
- blobToBinaryString: function (a, b) {
- var c = new FileReader;
- c.onload = function (a) {
- b(a.target.result)
- }, c.readAsBinaryString(a)
- },
- binaryStringToArrayBuffer: function (a) {
- for (var b = new Uint8Array(a.length), c = 0; c < a.length; c++) b[c] = 255 & a.charCodeAt(c);
- return b.buffer
- },
- randomToken: function () {
- return Math.random().toString(36).substr(2)
- },
- isSecure: function () {
- return "https:" === location.protocol
- }
- };
- b.exports = h
- }, {"./adapter": 1, "js-binarypack": 10}],
- 9: [function (a, b, c) {
- "use strict";
+ const i = a('./dataconnection');
+ e.inherits(d, f), d.prototype._initializeServerConnection = function () {
+ const a = this;
+ this.socket = new g(this.options.secure, this.options.host, this.options.port, this.options.path, this.options.key, this.options.wsport), this.socket.on('message', (b) => {
+ a._handleMessage(b);
+ }), this.socket.on('error', (b) => {
+ a._abort('socket-error', b);
+ }), this.socket.on('disconnected', () => {
+ a.disconnected || (a.emitError('network', 'Lost connection to server.'), a.disconnect());
+ }), this.socket.on('close', () => {
+ a.disconnected || a._abort('socket-closed', 'Underlying socket is already closed.');
+ });
+ }, d.prototype._retrieveId = function (a) {
+ const b = this;
+ const c = new XMLHttpRequest();
+ const d = this.options.secure ? 'https://' : 'http://';
- function d(a, b, c) {
- this.fn = a, this.context = b, this.once = c || !1
- }
- function e() {
- }
+ let f = `${d + this.options.host}:${this.options.port}${this.options.path}${this.options.key}/id`;
+ f += `?ts=${(new Date()).getTime()}${Math.random()}`, c.open('get', f, !0), c.onerror = function (a) {
+ e.error('Error retrieving ID', a);
+ let c = '';
+ b.options.path === '/' && b.options.host !== e.CLOUD_HOST && (c = ' If you passed in a `path` to your self-hosted PeerServer, you\'ll also need to pass in that same path when creating a new Peer.'), b._abort('server-error', `Could not get an ID from the server.${c}`);
+ }, c.onreadystatechange = function () {
+ if (c.readyState === 4) return c.status !== 200 ? void c.onerror() : void b._initialize(c.responseText);
+ }, c.send(null);
+ }, d.prototype._initialize = function (a) {
+ this.id = a, this.socket.start(this.id, this.options.token);
+ }, d.prototype._handleMessage = function (a) {
+ let b;
+ const c = a.type;
+ const d = a.payload;
+ const
+ f = a.src;
+ switch (c) {
+ case 'OPEN':
+ this.emit('open', this.id), this.open = !0;
+ break;
+ case 'ERROR':
+ this._abort('server-error', d.msg);
+ break;
+ case 'ID-TAKEN':
+ this._abort('unavailable-id', `ID \`${this.id}\` is taken`);
+ break;
+ case 'INVALID-KEY':
+ this._abort('invalid-key', `API KEY "${this.options.key}" is invalid`);
+ break;
+ case 'LEAVE':
+ e.log('Received leave message from', f), this._cleanupPeer(f);
+ break;
+ case 'EXPIRE':
+ this.emitError('peer-unavailable', `Could not connect to peer ${f}`);
+ break;
+ case 'OFFER':
+ var g = d.connectionId;
+ if (b = this.getConnection(f, g), b && (b.close(), e.warn('Offer received for existing Connection ID:', g)), d.type === 'media') {
+ b = new h(f, this, {
+ connectionId: g,
+ _payload: d,
+ metadata: d.metadata,
+ }), this._addConnection(f, b), this.emit('call', b);
+ } else {
+ if (d.type !== 'data') return void e.warn('Received malformed connection type:', d.type);
+ b = new i(f, this, {
+ connectionId: g,
+ _payload: d,
+ metadata: d.metadata,
+ label: d.label,
+ serialization: d.serialization,
+ reliable: d.reliable,
+ }), this._addConnection(f, b), this.emit('connection', b);
+ }
+ for (let j = this._getMessages(g), k = 0, l = j.length; k < l; k += 1) b.handleMessage(j[k]);
+ break;
+ default:
+ if (!d) return void e.warn(`You received a malformed message from ${f} of type ${c}`);
+ var m = d.connectionId;
+ b = this.getConnection(f, m), b && b.pc ? b.handleMessage(a) : m ? this._storeMessage(m, a) : e.warn('You received an unrecognized message:', a);
+ }
+ }, d.prototype._storeMessage = function (a, b) {
+ this._lostMessages[a] || (this._lostMessages[a] = []), this._lostMessages[a].push(b);
+ }, d.prototype._getMessages = function (a) {
+ const b = this._lostMessages[a];
+ return b ? (delete this._lostMessages[a], b) : [];
+ }, d.prototype.connect = function (a, b) {
+ if (this.disconnected) return e.warn('You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect, or call reconnect on this peer if you believe its ID to still be available.'), void this.emitError('disconnected', 'Cannot connect to new Peer after disconnecting from server.');
+ const c = new i(a, this, b);
+ return this._addConnection(a, c), c;
+ }, d.prototype.call = function (a, b, c) {
+ if (this.disconnected) return e.warn('You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect.'), void this.emitError('disconnected', 'Cannot connect to new Peer after disconnecting from server.');
+ if (!b) return void e.error('To call a peer, you must provide a stream from your browser\'s `getUserMedia`.');
+ c = c || {}, c._stream = b;
+ const d = new h(a, this, c);
+ return this._addConnection(a, d), d;
+ }, d.prototype._addConnection = function (a, b) {
+ this.connections[a] || (this.connections[a] = []), this.connections[a].push(b);
+ }, d.prototype.getConnection = function (a, b) {
+ const c = this.connections[a];
+ if (!c) return null;
+ for (let d = 0, e = c.length; d < e; d++) if (c[d].id === b) return c[d];
+ return null;
+ }, d.prototype._delayedAbort = function (a, b) {
+ const c = this;
+ e.setZeroTimeout(() => {
+ c._abort(a, b);
+ });
+ }, d.prototype._abort = function (a, b) {
+ e.error('Aborting!'), this._lastServerId ? this.disconnect() : this.destroy(), this.emitError(a, b);
+ }, d.prototype.emitError = function (a, b) {
+ e.error('Error:', b), typeof b === 'string' && (b = new Error(b)), b.type = a, this.emit('error', b);
+ }, d.prototype.destroy = function () {
+ this.destroyed || (this._cleanup(), this.disconnect(), this.destroyed = !0);
+ }, d.prototype._cleanup = function () {
+ if (this.connections) for (let a = Object.keys(this.connections), b = 0, c = a.length; b < c; b++) this._cleanupPeer(a[b]);
+ this.emit('close');
+ }, d.prototype._cleanupPeer = function (a) {
+ for (let b = this.connections[a], c = 0, d = b.length; c < d; c += 1) b[c].close();
+ }, d.prototype.disconnect = function () {
+ const a = this;
+ e.setZeroTimeout(() => {
+ a.disconnected || (a.disconnected = !0, a.open = !1, a.socket && a.socket.close(), a.emit('disconnected', a.id), a._lastServerId = a.id, a.id = null);
+ });
+ }, d.prototype.reconnect = function () {
+ if (this.disconnected && !this.destroyed) {
+ e.log(`Attempting reconnection to server with ID ${this._lastServerId}`), this.disconnected = !1, this._initializeServerConnection(), this._initialize(this._lastServerId);
+ } else {
+ if (this.destroyed) throw new Error('This peer cannot reconnect to the server. It has already been destroyed.');
+ if (this.disconnected || this.open) throw new Error(`Peer ${this.id} cannot reconnect because it is not disconnected from the server!`);
+ e.error('In a hurry? We\'re still trying to make the initial connection!');
+ }
+ }, d.prototype.listAllPeers = function (a) {
+ a = a || function () {
+ };
+ const b = this;
+ const c = new XMLHttpRequest();
+ const d = this.options.secure ? 'https://' : 'http://';
- e.prototype._events = void 0, e.prototype.listeners = function (a) {
- if (!this._events || !this._events[a]) return [];
- if (this._events[a].fn) return [this._events[a].fn];
- for (var b = 0, c = this._events[a].length, d = new Array(c); b < c; b++) d[b] = this._events[a][b].fn;
- return d
- }, e.prototype.emit = function (a, b, c, d, e, f) {
- if (!this._events || !this._events[a]) return !1;
- var g, h, i = this._events[a], j = arguments.length;
- if ("function" == typeof i.fn) {
- switch (i.once && this.removeListener(a, i.fn, !0), j) {
- case 1:
- return i.fn.call(i.context), !0;
- case 2:
- return i.fn.call(i.context, b), !0;
- case 3:
- return i.fn.call(i.context, b, c), !0;
- case 4:
- return i.fn.call(i.context, b, c, d), !0;
- case 5:
- return i.fn.call(i.context, b, c, d, e), !0;
- case 6:
- return i.fn.call(i.context, b, c, d, e, f), !0
- }
- for (h = 1, g = new Array(j - 1); h < j; h++) g[h - 1] = arguments[h];
- i.fn.apply(i.context, g)
- } else {
- var k, l = i.length;
- for (h = 0; h < l; h++) switch (i[h].once && this.removeListener(a, i[h].fn, !0), j) {
- case 1:
- i[h].fn.call(i[h].context);
- break;
- case 2:
- i[h].fn.call(i[h].context, b);
- break;
- case 3:
- i[h].fn.call(i[h].context, b, c);
- break;
- default:
- if (!g) for (k = 1, g = new Array(j - 1); k < j; k++) g[k - 1] = arguments[k];
- i[h].fn.apply(i[h].context, g)
- }
- }
- return !0
- }, e.prototype.on = function (a, b, c) {
- var e = new d(b, c || this);
- return this._events || (this._events = {}), this._events[a] ? this._events[a].fn ? this._events[a] = [this._events[a], e] : this._events[a].push(e) : this._events[a] = e, this
- }, e.prototype.once = function (a, b, c) {
- var e = new d(b, c || this, !0);
- return this._events || (this._events = {}), this._events[a] ? this._events[a].fn ? this._events[a] = [this._events[a], e] : this._events[a].push(e) : this._events[a] = e, this
- }, e.prototype.removeListener = function (a, b, c) {
- if (!this._events || !this._events[a]) return this;
- var d = this._events[a], e = [];
- if (b && (d.fn && (d.fn !== b || c && !d.once) && e.push(d), !d.fn)) for (var f = 0, g = d.length; f < g; f++) (d[f].fn !== b || c && !d[f].once) && e.push(d[f]);
- return e.length ? this._events[a] = 1 === e.length ? e[0] : e : delete this._events[a], this
- }, e.prototype.removeAllListeners = function (a) {
- return this._events ? (a ? delete this._events[a] : this._events = {}, this) : this
- }, e.prototype.off = e.prototype.removeListener, e.prototype.addListener = e.prototype.on, e.prototype.setMaxListeners = function () {
- return this
- }, e.EventEmitter = e, e.EventEmitter2 = e, e.EventEmitter3 = e, b.exports = e
- }, {}],
- 10: [function (a, b, c) {
- function d(a) {
- this.index = 0, this.dataBuffer = a, this.dataView = new Uint8Array(this.dataBuffer), this.length = this.dataBuffer.byteLength
- }
- function e() {
- this.bufferBuilder = new h
+ let f = `${d + this.options.host}:${this.options.port}${this.options.path}${this.options.key}/peers`;
+ f += `?ts=${(new Date()).getTime()}${Math.random()}`, c.open('get', f, !0), c.onerror = function (c) {
+ b._abort('server-error', 'Could not get peers from the server.'), a([]);
+ }, c.onreadystatechange = function () {
+ if (c.readyState === 4) {
+ if (c.status === 401) {
+ let d = '';
+ throw d = b.options.host !== e.CLOUD_HOST ? 'It looks like you\'re using the cloud server. You can email team@peerjs.com to enable peer listing for your API key.' : 'You need to enable `allow_discovery` on your self-hosted PeerServer to use this feature.', a([]), new Error(`It doesn't look like you have permission to list peers IDs. ${d}`);
+ }
+ a(c.status !== 200 ? [] : JSON.parse(c.responseText));
}
+ }, c.send(null);
+ }, b.exports = d;
+ }, {
+ './dataconnection': 2,
+ './mediaconnection': 4,
+ './socket': 7,
+ './util': 8,
+ eventemitter3: 9,
+ }],
+ 7: [function (a, b, c) {
+ function d(a, b, c, e, g, h) {
+ if (!(this instanceof d)) return new d(a, b, c, e, g, h);
+ h = h || c, f.call(this), this.disconnected = !1, this._queue = [];
+ const i = a ? 'https://' : 'http://';
- function f(a) {
- var b = a.charCodeAt(0);
- return b <= 2047 ? "00" : b <= 65535 ? "000" : b <= 2097151 ? "0000" : b <= 67108863 ? "00000" : "000000"
- }
- function g(a) {
- return a.length > 600 ? new Blob([a]).size : a.replace(/[^\u0000-\u007F]/g, f).length
- }
+ const j = a ? 'wss://' : 'ws://';
+ this._httpUrl = `${i + b}:${c}${e}${g}`, this._wsUrl = `${j + b}:${h}${e}peerjs?key=${g}`;
+ }
- var h = a("./bufferbuilder").BufferBuilder, i = a("./bufferbuilder").binaryFeatures, j = {
- unpack: function (a) {
- return new d(a).unpack()
- }, pack: function (a) {
- var b = new e;
- return b.pack(a), b.getBuffer()
- }
- };
- b.exports = j, d.prototype.unpack = function () {
- var a = this.unpack_uint8();
- if (a < 128) {
- return a
- }
- if ((224 ^ a) < 32) {
- return (224 ^ a) - 32
- }
- var b;
- if ((b = 160 ^ a) <= 15) return this.unpack_raw(b);
- if ((b = 176 ^ a) <= 15) return this.unpack_string(b);
- if ((b = 144 ^ a) <= 15) return this.unpack_array(b);
- if ((b = 128 ^ a) <= 15) return this.unpack_map(b);
- switch (a) {
- case 192:
- return null;
- case 193:
- return;
- case 194:
- return !1;
- case 195:
- return !0;
- case 202:
- return this.unpack_float();
- case 203:
- return this.unpack_double();
- case 204:
- return this.unpack_uint8();
- case 205:
- return this.unpack_uint16();
- case 206:
- return this.unpack_uint32();
- case 207:
- return this.unpack_uint64();
- case 208:
- return this.unpack_int8();
- case 209:
- return this.unpack_int16();
- case 210:
- return this.unpack_int32();
- case 211:
- return this.unpack_int64();
- case 212:
- case 213:
- case 214:
- case 215:
- return;
- case 216:
- return b = this.unpack_uint16(), this.unpack_string(b);
- case 217:
- return b = this.unpack_uint32(), this.unpack_string(b);
- case 218:
- return b = this.unpack_uint16(), this.unpack_raw(b);
- case 219:
- return b = this.unpack_uint32(), this.unpack_raw(b);
- case 220:
- return b = this.unpack_uint16(), this.unpack_array(b);
- case 221:
- return b = this.unpack_uint32(), this.unpack_array(b);
- case 222:
- return b = this.unpack_uint16(), this.unpack_map(b);
- case 223:
- return b = this.unpack_uint32(), this.unpack_map(b)
- }
- }, d.prototype.unpack_uint8 = function () {
- var a = 255 & this.dataView[this.index];
- return this.index++, a
- }, d.prototype.unpack_uint16 = function () {
- var a = this.read(2), b = 256 * (255 & a[0]) + (255 & a[1]);
- return this.index += 2, b
- }, d.prototype.unpack_uint32 = function () {
- var a = this.read(4), b = 256 * (256 * (256 * a[0] + a[1]) + a[2]) + a[3];
- return this.index += 4, b
- }, d.prototype.unpack_uint64 = function () {
- var a = this.read(8),
- b = 256 * (256 * (256 * (256 * (256 * (256 * (256 * a[0] + a[1]) + a[2]) + a[3]) + a[4]) + a[5]) + a[6]) + a[7];
- return this.index += 8, b
- }, d.prototype.unpack_int8 = function () {
- var a = this.unpack_uint8();
- return a < 128 ? a : a - 256
- }, d.prototype.unpack_int16 = function () {
- var a = this.unpack_uint16();
- return a < 32768 ? a : a - 65536
- }, d.prototype.unpack_int32 = function () {
- var a = this.unpack_uint32();
- return a < Math.pow(2, 31) ? a : a - Math.pow(2, 32)
- }, d.prototype.unpack_int64 = function () {
- var a = this.unpack_uint64();
- return a < Math.pow(2, 63) ? a : a - Math.pow(2, 64)
- }, d.prototype.unpack_raw = function (a) {
- if (this.length < this.index + a) throw new Error("BinaryPackFailure: index is out of range " + this.index + " " + a + " " + this.length);
- var b = this.dataBuffer.slice(this.index, this.index + a);
- return this.index += a, b
- }, d.prototype.unpack_string = function (a) {
- for (var b, c, d = this.read(a), e = 0, f = ""; e < a;) b = d[e], b < 128 ? (f += String.fromCharCode(b), e++) : (192 ^ b) < 32 ? (c = (192 ^ b) << 6 | 63 & d[e + 1], f += String.fromCharCode(c), e += 2) : (c = (15 & b) << 12 | (63 & d[e + 1]) << 6 | 63 & d[e + 2], f += String.fromCharCode(c), e += 3);
- return this.index += a, f
- }, d.prototype.unpack_array = function (a) {
- for (var b = new Array(a), c = 0; c < a; c++) b[c] = this.unpack();
- return b
- }, d.prototype.unpack_map = function (a) {
- for (var b = {}, c = 0; c < a; c++) {
- var d = this.unpack(), e = this.unpack();
- b[d] = e
- }
- return b
- }, d.prototype.unpack_float = function () {
- var a = this.unpack_uint32(), b = a >> 31, c = (a >> 23 & 255) - 127, d = 8388607 & a | 8388608;
- return (0 == b ? 1 : -1) * d * Math.pow(2, c - 23)
- }, d.prototype.unpack_double = function () {
- var a = this.unpack_uint32(), b = this.unpack_uint32(), c = a >> 31, d = (a >> 20 & 2047) - 1023,
- e = 1048575 & a | 1048576, f = e * Math.pow(2, d - 20) + b * Math.pow(2, d - 52);
- return (0 == c ? 1 : -1) * f
- }, d.prototype.read = function (a) {
- var b = this.index;
- if (b + a <= this.length) return this.dataView.subarray(b, b + a);
- throw new Error("BinaryPackFailure: read index out of range")
- }, e.prototype.getBuffer = function () {
- return this.bufferBuilder.getBuffer()
- }, e.prototype.pack = function (a) {
- var b = typeof a;
- if ("string" == b) this.pack_string(a); else if ("number" == b) Math.floor(a) === a ? this.pack_integer(a) : this.pack_double(a); else if ("boolean" == b) !0 === a ? this.bufferBuilder.append(195) : !1 === a && this.bufferBuilder.append(194); else if ("undefined" == b) this.bufferBuilder.append(192); else {
- if ("object" != b) throw new Error('Type "' + b + '" not yet supported');
- if (null === a) this.bufferBuilder.append(192); else {
- var c = a.constructor;
- if (c == Array) this.pack_array(a); else if (c == Blob || c == File) this.pack_bin(a); else if (c == ArrayBuffer) i.useArrayBufferView ? this.pack_bin(new Uint8Array(a)) : this.pack_bin(a); else if ("BYTES_PER_ELEMENT" in a) i.useArrayBufferView ? this.pack_bin(new Uint8Array(a.buffer)) : this.pack_bin(a.buffer); else if (c == Object) this.pack_object(a); else if (c == Date) this.pack_string(a.toString()); else {
- if ("function" != typeof a.toBinaryPack) throw new Error('Type "' + c.toString() + '" not yet supported');
- this.bufferBuilder.append(a.toBinaryPack())
- }
- }
- }
- this.bufferBuilder.flush()
- }, e.prototype.pack_bin = function (a) {
- var b = a.length || a.byteLength || a.size;
- if (b <= 15) this.pack_uint8(160 + b); else if (b <= 65535) this.bufferBuilder.append(218), this.pack_uint16(b); else {
- if (!(b <= 4294967295)) throw new Error("Invalid length");
- this.bufferBuilder.append(219), this.pack_uint32(b)
- }
- this.bufferBuilder.append(a)
- }, e.prototype.pack_string = function (a) {
- var b = g(a);
- if (b <= 15) this.pack_uint8(176 + b); else if (b <= 65535) this.bufferBuilder.append(216), this.pack_uint16(b); else {
- if (!(b <= 4294967295)) throw new Error("Invalid length");
- this.bufferBuilder.append(217), this.pack_uint32(b)
- }
- this.bufferBuilder.append(a)
- }, e.prototype.pack_array = function (a) {
- var b = a.length;
- if (b <= 15) this.pack_uint8(144 + b); else if (b <= 65535) this.bufferBuilder.append(220), this.pack_uint16(b); else {
- if (!(b <= 4294967295)) throw new Error("Invalid length");
- this.bufferBuilder.append(221), this.pack_uint32(b)
- }
- for (var c = 0; c < b; c++) this.pack(a[c])
- }, e.prototype.pack_integer = function (a) {
- if (-32 <= a && a <= 127) this.bufferBuilder.append(255 & a); else if (0 <= a && a <= 255) this.bufferBuilder.append(204), this.pack_uint8(a); else if (-128 <= a && a <= 127) this.bufferBuilder.append(208), this.pack_int8(a); else if (0 <= a && a <= 65535) this.bufferBuilder.append(205), this.pack_uint16(a); else if (-32768 <= a && a <= 32767) this.bufferBuilder.append(209), this.pack_int16(a); else if (0 <= a && a <= 4294967295) this.bufferBuilder.append(206), this.pack_uint32(a); else if (-2147483648 <= a && a <= 2147483647) this.bufferBuilder.append(210), this.pack_int32(a); else if (-0x8000000000000000 <= a && a <= 0x8000000000000000) this.bufferBuilder.append(211), this.pack_int64(a); else {
- if (!(0 <= a && a <= 0x10000000000000000)) throw new Error("Invalid integer");
- this.bufferBuilder.append(207), this.pack_uint64(a)
- }
- }, e.prototype.pack_double = function (a) {
- var b = 0;
- a < 0 && (b = 1, a = -a);
- var c = Math.floor(Math.log(a) / Math.LN2), d = a / Math.pow(2, c) - 1, e = Math.floor(d * Math.pow(2, 52)),
- f = Math.pow(2, 32), g = b << 31 | c + 1023 << 20 | e / f & 1048575, h = e % f;
- this.bufferBuilder.append(203), this.pack_int32(g), this.pack_int32(h)
- }, e.prototype.pack_object = function (a) {
- var b = Object.keys(a), c = b.length;
- if (c <= 15) this.pack_uint8(128 + c); else if (c <= 65535) this.bufferBuilder.append(222), this.pack_uint16(c); else {
- if (!(c <= 4294967295)) throw new Error("Invalid length");
- this.bufferBuilder.append(223), this.pack_uint32(c)
- }
- for (var d in a) a.hasOwnProperty(d) && (this.pack(d), this.pack(a[d]))
- }, e.prototype.pack_uint8 = function (a) {
- this.bufferBuilder.append(a)
- }, e.prototype.pack_uint16 = function (a) {
- this.bufferBuilder.append(a >> 8), this.bufferBuilder.append(255 & a)
- }, e.prototype.pack_uint32 = function (a) {
- var b = 4294967295 & a;
- this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b)
- }, e.prototype.pack_uint64 = function (a) {
- var b = a / Math.pow(2, 32), c = a % Math.pow(2, 32);
- this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b), this.bufferBuilder.append((4278190080 & c) >>> 24), this.bufferBuilder.append((16711680 & c) >>> 16), this.bufferBuilder.append((65280 & c) >>> 8), this.bufferBuilder.append(255 & c)
- }, e.prototype.pack_int8 = function (a) {
- this.bufferBuilder.append(255 & a)
- }, e.prototype.pack_int16 = function (a) {
- this.bufferBuilder.append((65280 & a) >> 8), this.bufferBuilder.append(255 & a)
- }, e.prototype.pack_int32 = function (a) {
- this.bufferBuilder.append(a >>> 24 & 255), this.bufferBuilder.append((16711680 & a) >>> 16), this.bufferBuilder.append((65280 & a) >>> 8), this.bufferBuilder.append(255 & a)
- }, e.prototype.pack_int64 = function (a) {
- var b = Math.floor(a / Math.pow(2, 32)), c = a % Math.pow(2, 32);
- this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b), this.bufferBuilder.append((4278190080 & c) >>> 24), this.bufferBuilder.append((16711680 & c) >>> 16), this.bufferBuilder.append((65280 & c) >>> 8), this.bufferBuilder.append(255 & c)
+ const e = a('./util');
+ var
+ f = a('eventemitter3');
+ e.inherits(d, f), d.prototype.start = function (a, b) {
+ this.id = a, this._httpUrl += `/${a}/${b}`, this._wsUrl += `&id=${a}&token=${b}`, this._startXhrStream(), this._startWebSocket();
+ }, d.prototype._startWebSocket = function (a) {
+ const b = this;
+ this._socket || (this._socket = new WebSocket(this._wsUrl), this._socket.onmessage = function (a) {
+ try {
+ var c = JSON.parse(a.data);
+ } catch (b) {
+ return void e.log('Invalid server message', a.data);
+ }
+ b.emit('message', c);
+ }, this._socket.onclose = function (a) {
+ e.log('Socket closed.'), b.disconnected = !0, b.emit('disconnected');
+ }, this._socket.onopen = function () {
+ b._timeout && (clearTimeout(b._timeout), setTimeout(() => {
+ b._http.abort(), b._http = null;
+ }, 5e3)), b._sendQueuedMessages(), e.log('Socket open');
+ });
+ }, d.prototype._startXhrStream = function (a) {
+ try {
+ const b = this;
+ this._http = new XMLHttpRequest(), this._http._index = 1, this._http._streamIndex = a || 0, this._http.open('post', `${this._httpUrl}/id?i=${this._http._streamIndex}`, !0), this._http.onerror = function () {
+ clearTimeout(b._timeout), b.emit('disconnected');
+ }, this._http.onreadystatechange = function () {
+ this.readyState == 2 && this.old ? (this.old.abort(), delete this.old) : this.readyState > 2 && this.status === 200 && this.responseText && b._handleStream(this);
+ }, this._http.send(null), this._setHTTPTimeout();
+ } catch (a) {
+ e.log('XMLHttpRequest not available; defaulting to WebSockets');
+ }
+ }, d.prototype._handleStream = function (a) {
+ const b = a.responseText.split('\n');
+ if (a._buffer) {
+ for (; a._buffer.length > 0;) {
+ const c = a._buffer.shift();
+ let
+ d = b[c];
+ try {
+ d = JSON.parse(d);
+ } catch (b) {
+ a._buffer.shift(c);
+ break;
+ }
+ this.emit('message', d);
}
- }, {"./bufferbuilder": 11}],
- 11: [function (a, b, c) {
- function d() {
- this._pieces = [], this._parts = []
+ }
+ let f = b[a._index];
+ if (f) {
+ if (a._index += 1, a._index === b.length) {
+ a._buffer || (a._buffer = []), a._buffer.push(a._index - 1);
+ } else {
+ try {
+ f = JSON.parse(f);
+ } catch (a) {
+ return void e.log('Invalid server message', f);
+ }
+ this.emit('message', f);
+ }
+ }
+ }, d.prototype._setHTTPTimeout = function () {
+ const a = this;
+ this._timeout = setTimeout(() => {
+ const b = a._http;
+ a._wsOpen() ? b.abort() : (a._startXhrStream(b._streamIndex + 1), a._http.old = b);
+ }, 25e3);
+ }, d.prototype._wsOpen = function () {
+ return this._socket && this._socket.readyState == 1;
+ }, d.prototype._sendQueuedMessages = function () {
+ for (let a = 0, b = this._queue.length; a < b; a += 1) this.send(this._queue[a]);
+ }, d.prototype.send = function (a) {
+ if (!this.disconnected) {
+ if (!this.id) return void this._queue.push(a);
+ if (!a.type) return void this.emit('error', 'Invalid message');
+ const b = JSON.stringify(a);
+ if (this._wsOpen()) {
+ this._socket.send(b);
+ } else {
+ const c = new XMLHttpRequest();
+
+
+ const d = `${this._httpUrl}/${a.type.toLowerCase()}`;
+ c.open('post', d, !0), c.setRequestHeader('Content-Type', 'application/json'), c.send(b);
}
+ }
+ }, d.prototype.close = function () {
+ !this.disconnected && this._wsOpen() && (this._socket.close(), this.disconnected = !0);
+ }, b.exports = d;
+ }, {
+ './util': 8,
+ eventemitter3: 9
+ }],
+ 8: [function (a, b, c) {
+ const d = { iceServers: [{ url: 'stun:stun.l.google.com:19302' }] };
+ let e = 1;
+ const f = a('js-binarypack');
- var e = {};
- e.useBlobBuilder = function () {
+
+ const g = a('./adapter').RTCPeerConnection;
+ var
+ h = {
+ noop() {
+ },
+ CLOUD_HOST: '0.peerjs.com',
+ CLOUD_PORT: 9e3,
+ chunkedBrowsers: { Chrome: 1 },
+ chunkedMTU: 16300,
+ logLevel: 0,
+ setLogLevel(a) {
+ const b = parseInt(a, 10);
+ isNaN(parseInt(a, 10)) ? h.logLevel = a ? 3 : 0 : h.logLevel = b, h.log = h.warn = h.error = h.noop, h.logLevel > 0 && (h.error = h._printWith('ERROR')), h.logLevel > 1 && (h.warn = h._printWith('WARNING')), h.logLevel > 2 && (h.log = h._print);
+ },
+ setLogFunction(a) {
+ a.constructor !== Function ? h.warn('The log function you passed in is not a function. Defaulting to regular logs.') : h._print = a;
+ },
+ _printWith(a) {
+ return function () {
+ const b = Array.prototype.slice.call(arguments);
+ b.unshift(a), h._print(...b);
+ };
+ },
+ _print() {
+ let a = !1;
+ const
+ b = Array.prototype.slice.call(arguments);
+ b.unshift('PeerJS: ');
+ for (let c = 0, d = b.length; c < d; c++) b[c] instanceof Error && (b[c] = `(${b[c].name}) ${b[c].message}`, a = !0);
+ a ? console.error(...b) : console.log(...b);
+ },
+ defaultConfig: d,
+ browser: (function () {
+ return window.mozRTCPeerConnection ? 'Firefox' : window.webkitRTCPeerConnection ? 'Chrome' : window.RTCPeerConnection ? 'Supported' : 'Unsupported';
+ }()),
+ supports: (function () {
+ if (void 0 === g) return {};
+ let a;
+ let b;
+ let c = !0;
+ let e = !0;
+ let f = !1;
+ let h = !1;
+ const
+ i = !!window.webkitRTCPeerConnection;
+ try {
+ a = new g(d, { optional: [{ RtpDataChannels: !0 }] });
+ } catch (a) {
+ c = !1, e = !1;
+ }
+ if (c) {
try {
- return new Blob([]), !1
+ b = a.createDataChannel('_PEERJSTEST');
} catch (a) {
- return !0
+ c = !1;
}
- }(), e.useArrayBufferView = !e.useBlobBuilder && function () {
+ }
+ if (c) {
try {
- return 0 === new Blob([new Uint8Array([])]).size
+ b.binaryType = 'blob', f = !0;
} catch (a) {
- return !0
- }
- }(), b.exports.binaryFeatures = e;
- var f = b.exports.BlobBuilder;
- "undefined" != typeof window && (f = b.exports.BlobBuilder = window.WebKitBlobBuilder || window.MozBlobBuilder || window.MSBlobBuilder || window.BlobBuilder), d.prototype.append = function (a) {
- "number" == typeof a ? this._pieces.push(a) : (this.flush(), this._parts.push(a))
- }, d.prototype.flush = function () {
- if (this._pieces.length > 0) {
- var a = new Uint8Array(this._pieces);
- e.useArrayBufferView || (a = a.buffer), this._parts.push(a), this._pieces = []
}
- }, d.prototype.getBuffer = function () {
- if (this.flush(), e.useBlobBuilder) {
- for (var a = new f, b = 0, c = this._parts.length; b < c; b++) a.append(this._parts[b]);
- return a.getBlob()
+ const j = new g(d, {});
+ try {
+ h = j.createDataChannel('_PEERJSRELIABLETEST', {}).reliable;
+ } catch (a) {
}
- return new Blob(this._parts)
- }, b.exports.BufferBuilder = d
- }, {}],
- 12: [function (a, b, c) {
- function d(a, b) {
- if (!(this instanceof d)) return new d(a);
- this._dc = a, e.debug = b, this._outgoing = {}, this._incoming = {}, this._received = {}, this._window = 1e3, this._mtu = 500, this._interval = 0, this._count = 0, this._queue = [], this._setupDC()
+ j.close();
+ }
+ return e && (e = !!a.addStream), a && a.close(), {
+ audioVideo: e,
+ data: c,
+ binaryBlob: f,
+ binary: h,
+ reliable: h,
+ sctp: h,
+ onnegotiationneeded: i,
+ };
+ }()),
+ validateId(a) {
+ return !a || /^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(a);
+ },
+ validateKey(a) {
+ return !a || /^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(a);
+ },
+ debug: !1,
+ inherits(a, b) {
+ a.super_ = b, a.prototype = Object.create(b.prototype, {
+ constructor: {
+ value: a,
+ enumerable: !1,
+ writable: !0,
+ configurable: !0,
+ },
+ });
+ },
+ extend(a, b) {
+ for (const c in b) b.hasOwnProperty(c) && (a[c] = b[c]);
+ return a;
+ },
+ pack: f.pack,
+ unpack: f.unpack,
+ log() {
+ if (h.debug) {
+ let a = !1;
+ const
+ b = Array.prototype.slice.call(arguments);
+ b.unshift('PeerJS: ');
+ for (let c = 0, d = b.length; c < d; c++) b[c] instanceof Error && (b[c] = `(${b[c].name}) ${b[c].message}`, a = !0);
+ a ? console.error(...b) : console.log(...b);
+ }
+ },
+ setZeroTimeout: (function (a) {
+ function b(b) {
+ d.push(b), a.postMessage(e, '*');
+ }
+
+ function c(b) {
+ b.source == a && b.data == e && (b.stopPropagation && b.stopPropagation(), d.length && d.shift()());
+ }
+
+ var d = [];
+ var
+ e = 'zero-timeout-message';
+ return a.addEventListener ? a.addEventListener('message', c, !0) : a.attachEvent && a.attachEvent('onmessage', c), b;
+ }(window)),
+ chunk(a) {
+ for (var b = [], c = a.size, d = index = 0, f = Math.ceil(c / h.chunkedMTU); d < c;) {
+ const g = Math.min(c, d + h.chunkedMTU);
+ const i = a.slice(d, g);
+
+
+ const j = {
+ __peerData: e,
+ n: index,
+ data: i,
+ total: f,
+ };
+ b.push(j), d = g, index += 1;
+ }
+ return e += 1, b;
+ },
+ blobToArrayBuffer(a, b) {
+ const c = new FileReader();
+ c.onload = function (a) {
+ b(a.target.result);
+ }, c.readAsArrayBuffer(a);
+ },
+ blobToBinaryString(a, b) {
+ const c = new FileReader();
+ c.onload = function (a) {
+ b(a.target.result);
+ }, c.readAsBinaryString(a);
+ },
+ binaryStringToArrayBuffer(a) {
+ for (var b = new Uint8Array(a.length), c = 0; c < a.length; c++) b[c] = 255 & a.charCodeAt(c);
+ return b.buffer;
+ },
+ randomToken() {
+ return Math.random()
+ .toString(36)
+ .substr(2);
+ },
+ isSecure() {
+ return location.protocol === 'https:';
+ },
+ };
+ b.exports = h;
+ }, {
+ './adapter': 1,
+ 'js-binarypack': 10
+ }],
+ 9: [function (a, b, c) {
+ function d(a, b, c) {
+ this.fn = a, this.context = b, this.once = c || !1;
+ }
+
+ function e() {
+ }
+
+ e.prototype._events = void 0, e.prototype.listeners = function (a) {
+ if (!this._events || !this._events[a]) return [];
+ if (this._events[a].fn) return [this._events[a].fn];
+ for (var b = 0, c = this._events[a].length, d = new Array(c); b < c; b++) d[b] = this._events[a][b].fn;
+ return d;
+ }, e.prototype.emit = function (a, b, c, d, e, f) {
+ if (!this._events || !this._events[a]) return !1;
+ let g;
+ let h;
+ const i = this._events[a];
+ const
+ j = arguments.length;
+ if (typeof i.fn === 'function') {
+ switch (i.once && this.removeListener(a, i.fn, !0), j) {
+ case 1:
+ return i.fn.call(i.context), !0;
+ case 2:
+ return i.fn.call(i.context, b), !0;
+ case 3:
+ return i.fn.call(i.context, b, c), !0;
+ case 4:
+ return i.fn.call(i.context, b, c, d), !0;
+ case 5:
+ return i.fn.call(i.context, b, c, d, e), !0;
+ case 6:
+ return i.fn.call(i.context, b, c, d, e, f), !0;
+ }
+ for (h = 1, g = new Array(j - 1); h < j; h++) g[h - 1] = arguments[h];
+ i.fn.apply(i.context, g);
+ } else {
+ let k;
+ const
+ l = i.length;
+ for (h = 0; h < l; h++) {
+ switch (i[h].once && this.removeListener(a, i[h].fn, !0), j) {
+ case 1:
+ i[h].fn.call(i[h].context);
+ break;
+ case 2:
+ i[h].fn.call(i[h].context, b);
+ break;
+ case 3:
+ i[h].fn.call(i[h].context, b, c);
+ break;
+ default:
+ if (!g) for (k = 1, g = new Array(j - 1); k < j; k++) g[k - 1] = arguments[k];
+ i[h].fn.apply(i[h].context, g);
+ }
}
+ }
+ return !0;
+ }, e.prototype.on = function (a, b, c) {
+ const e = new d(b, c || this);
+ return this._events || (this._events = {}), this._events[a] ? this._events[a].fn ? this._events[a] = [this._events[a], e] : this._events[a].push(e) : this._events[a] = e, this;
+ }, e.prototype.once = function (a, b, c) {
+ const e = new d(b, c || this, !0);
+ return this._events || (this._events = {}), this._events[a] ? this._events[a].fn ? this._events[a] = [this._events[a], e] : this._events[a].push(e) : this._events[a] = e, this;
+ }, e.prototype.removeListener = function (a, b, c) {
+ if (!this._events || !this._events[a]) return this;
+ const d = this._events[a];
+ const
+ e = [];
+ if (b && (d.fn && (d.fn !== b || c && !d.once) && e.push(d), !d.fn)) for (let f = 0, g = d.length; f < g; f++) (d[f].fn !== b || c && !d[f].once) && e.push(d[f]);
+ return e.length ? this._events[a] = e.length === 1 ? e[0] : e : delete this._events[a], this;
+ }, e.prototype.removeAllListeners = function (a) {
+ return this._events ? (a ? delete this._events[a] : this._events = {}, this) : this;
+ }, e.prototype.off = e.prototype.removeListener, e.prototype.addListener = e.prototype.on, e.prototype.setMaxListeners = function () {
+ return this;
+ }, e.EventEmitter = e, e.EventEmitter2 = e, e.EventEmitter3 = e, b.exports = e;
+ }, {}],
+ 10: [function (a, b, c) {
+ function d(a) {
+ this.index = 0, this.dataBuffer = a, this.dataView = new Uint8Array(this.dataBuffer), this.length = this.dataBuffer.byteLength;
+ }
- var e = a("./util");
- d.prototype.send = function (a) {
- var b = e.pack(a);
- if (b.size < this._mtu) return void this._handleSend(["no", b]);
- this._outgoing[this._count] = {
- ack: 0,
- chunks: this._chunk(b)
- }, e.debug && (this._outgoing[this._count].timer = new Date), this._sendWindowedChunks(this._count), this._count += 1
- }, d.prototype._setupInterval = function () {
- var a = this;
- this._timeout = setInterval(function () {
- var b = a._queue.shift();
- if (b._multiple) for (var c = 0, d = b.length; c < d; c += 1) a._intervalSend(b[c]); else a._intervalSend(b)
- }, this._interval)
- }, d.prototype._intervalSend = function (a) {
- var b = this;
- a = e.pack(a), e.blobToBinaryString(a, function (a) {
- b._dc.send(a)
- }), 0 === b._queue.length && (clearTimeout(b._timeout), b._timeout = null)
- }, d.prototype._processAcks = function () {
- for (var a in this._outgoing) this._outgoing.hasOwnProperty(a) && this._sendWindowedChunks(a)
- }, d.prototype._handleSend = function (a) {
- for (var b = !0, c = 0, d = this._queue.length; c < d; c += 1) {
- var e = this._queue[c];
- e === a ? b = !1 : e._multiple && -1 !== e.indexOf(a) && (b = !1)
- }
- b && (this._queue.push(a), this._timeout || this._setupInterval())
- }, d.prototype._setupDC = function () {
- var a = this;
- this._dc.onmessage = function (b) {
- var c = b.data;
- if (c.constructor === String) {
- var d = e.binaryStringToArrayBuffer(c);
- c = e.unpack(d), a._handleMessage(c)
- }
- }
- }, d.prototype._handleMessage = function (a) {
- var b, c = a[1], d = this._incoming[c], f = this._outgoing[c];
- switch (a[0]) {
- case"no":
- var g = c;
- g && this.onmessage(e.unpack(g));
- break;
- case"end":
- if (b = d, this._received[c] = a[2], !b) break;
- this._ack(c);
- break;
- case"ack":
- if (b = f) {
- var h = a[2];
- b.ack = Math.max(h, b.ack), b.ack >= b.chunks.length ? (e.log("Time: ", new Date - b.timer), delete this._outgoing[c]) : this._processAcks()
- }
- break;
- case"chunk":
- if (!(b = d)) {
- if (!0 === this._received[c]) break;
- b = {ack: ["ack", c, 0], chunks: []}, this._incoming[c] = b
- }
- var i = a[2], j = a[3];
- b.chunks[i] = new Uint8Array(j), i === b.ack[2] && this._calculateNextAck(c), this._ack(c);
- break;
- default:
- this._handleSend(a)
- }
- }, d.prototype._chunk = function (a) {
- for (var b = [], c = a.size, d = 0; d < c;) {
- var f = Math.min(c, d + this._mtu), g = a.slice(d, f), h = {payload: g};
- b.push(h), d = f
- }
- return e.log("Created", b.length, "chunks."), b
- }, d.prototype._ack = function (a) {
- var b = this._incoming[a].ack;
- this._received[a] === b[2] && (this._complete(a), this._received[a] = !0), this._handleSend(b)
- }, d.prototype._calculateNextAck = function (a) {
- for (var b = this._incoming[a], c = b.chunks, d = 0, e = c.length; d < e; d += 1) if (void 0 === c[d]) return void (b.ack[2] = d);
- b.ack[2] = c.length
- }, d.prototype._sendWindowedChunks = function (a) {
- e.log("sendWindowedChunks for: ", a);
- for (var b = this._outgoing[a], c = b.chunks, d = [], f = Math.min(b.ack + this._window, c.length), g = b.ack; g < f; g += 1) c[g].sent && g !== b.ack || (c[g].sent = !0, d.push(["chunk", a, g, c[g].payload]));
- b.ack + this._window >= c.length && d.push(["end", a, c.length]), d._multiple = !0, this._handleSend(d)
- }, d.prototype._complete = function (a) {
- e.log("Completed called for", a);
- var b = this, c = this._incoming[a].chunks, d = new Blob(c);
- e.blobToArrayBuffer(d, function (a) {
- b.onmessage(e.unpack(a))
- }), delete this._incoming[a]
- }, d.higherBandwidthSDP = function (a) {
- var b = navigator.appVersion.match(/Chrome\/(.*?) /);
- if (b && (b = parseInt(b[1].split(".").shift())) < 31) {
- var c = a.split("b=AS:30");
- if (c.length > 1) return c[0] + "b=AS:102400" + c[1]
- }
- return a
- }, d.prototype.onmessage = function (a) {
- }, b.exports.Reliable = d
- }, {"./util": 13}],
- 13: [function (a, b, c) {
- var d = a("js-binarypack"), e = {
- debug: !1, inherits: function (a, b) {
- a.super_ = b, a.prototype = Object.create(b.prototype, {
- constructor: {
- value: a,
- enumerable: !1,
- writable: !0,
- configurable: !0
- }
- })
- }, extend: function (a, b) {
- for (var c in b) b.hasOwnProperty(c) && (a[c] = b[c]);
- return a
- }, pack: d.pack, unpack: d.unpack, log: function () {
- if (e.debug) {
- for (var a = [], b = 0; b < arguments.length; b++) a[b] = arguments[b];
- a.unshift("Reliable: "), console.log.apply(console, a)
- }
- }, setZeroTimeout: function (a) {
- function b(b) {
- d.push(b), a.postMessage(e, "*")
- }
+ function e() {
+ this.bufferBuilder = new h();
+ }
- function c(b) {
- b.source == a && b.data == e && (b.stopPropagation && b.stopPropagation(), d.length && d.shift()())
- }
+ function f(a) {
+ const b = a.charCodeAt(0);
+ return b <= 2047 ? '00' : b <= 65535 ? '000' : b <= 2097151 ? '0000' : b <= 67108863 ? '00000' : '000000';
+ }
- var d = [], e = "zero-timeout-message";
- return a.addEventListener ? a.addEventListener("message", c, !0) : a.attachEvent && a.attachEvent("onmessage", c), b
- }(this), blobToArrayBuffer: function (a, b) {
- var c = new FileReader;
- c.onload = function (a) {
- b(a.target.result)
- }, c.readAsArrayBuffer(a)
- }, blobToBinaryString: function (a, b) {
- var c = new FileReader;
- c.onload = function (a) {
- b(a.target.result)
- }, c.readAsBinaryString(a)
- }, binaryStringToArrayBuffer: function (a) {
- for (var b = new Uint8Array(a.length), c = 0; c < a.length; c++) b[c] = 255 & a.charCodeAt(c);
- return b.buffer
- }, randomToken: function () {
- return Math.random().toString(36).substr(2)
- }
- };
- b.exports = e
- }, {"js-binarypack": 10}]
-}, {}, [3]);
+ function g(a) {
+ return a.length > 600 ? new Blob([a]).size : a.replace(/[^\u0000-\u007F]/g, f).length;
+ }
+
+ var h = a('./bufferbuilder').BufferBuilder;
+ const i = a('./bufferbuilder').binaryFeatures;
+ const
+ j = {
+ unpack(a) {
+ return new d(a).unpack();
+ },
+ pack(a) {
+ const b = new e();
+ return b.pack(a), b.getBuffer();
+ },
+ };
+ b.exports = j, d.prototype.unpack = function () {
+ const a = this.unpack_uint8();
+ if (a < 128) {
+ return a;
+ }
+ if ((224 ^ a) < 32) {
+ return (224 ^ a) - 32;
+ }
+ let b;
+ if ((b = 160 ^ a) <= 15) return this.unpack_raw(b);
+ if ((b = 176 ^ a) <= 15) return this.unpack_string(b);
+ if ((b = 144 ^ a) <= 15) return this.unpack_array(b);
+ if ((b = 128 ^ a) <= 15) return this.unpack_map(b);
+ switch (a) {
+ case 192:
+ return null;
+ case 193:
+ return;
+ case 194:
+ return !1;
+ case 195:
+ return !0;
+ case 202:
+ return this.unpack_float();
+ case 203:
+ return this.unpack_double();
+ case 204:
+ return this.unpack_uint8();
+ case 205:
+ return this.unpack_uint16();
+ case 206:
+ return this.unpack_uint32();
+ case 207:
+ return this.unpack_uint64();
+ case 208:
+ return this.unpack_int8();
+ case 209:
+ return this.unpack_int16();
+ case 210:
+ return this.unpack_int32();
+ case 211:
+ return this.unpack_int64();
+ case 212:
+ case 213:
+ case 214:
+ case 215:
+ return;
+ case 216:
+ return b = this.unpack_uint16(), this.unpack_string(b);
+ case 217:
+ return b = this.unpack_uint32(), this.unpack_string(b);
+ case 218:
+ return b = this.unpack_uint16(), this.unpack_raw(b);
+ case 219:
+ return b = this.unpack_uint32(), this.unpack_raw(b);
+ case 220:
+ return b = this.unpack_uint16(), this.unpack_array(b);
+ case 221:
+ return b = this.unpack_uint32(), this.unpack_array(b);
+ case 222:
+ return b = this.unpack_uint16(), this.unpack_map(b);
+ case 223:
+ return b = this.unpack_uint32(), this.unpack_map(b);
+ }
+ }, d.prototype.unpack_uint8 = function () {
+ const a = 255 & this.dataView[this.index];
+ return this.index++, a;
+ }, d.prototype.unpack_uint16 = function () {
+ const a = this.read(2);
+ const
+ b = 256 * (255 & a[0]) + (255 & a[1]);
+ return this.index += 2, b;
+ }, d.prototype.unpack_uint32 = function () {
+ const a = this.read(4);
+ const
+ b = 256 * (256 * (256 * a[0] + a[1]) + a[2]) + a[3];
+ return this.index += 4, b;
+ }, d.prototype.unpack_uint64 = function () {
+ const a = this.read(8);
+
+
+ const b = 256 * (256 * (256 * (256 * (256 * (256 * (256 * a[0] + a[1]) + a[2]) + a[3]) + a[4]) + a[5]) + a[6]) + a[7];
+ return this.index += 8, b;
+ }, d.prototype.unpack_int8 = function () {
+ const a = this.unpack_uint8();
+ return a < 128 ? a : a - 256;
+ }, d.prototype.unpack_int16 = function () {
+ const a = this.unpack_uint16();
+ return a < 32768 ? a : a - 65536;
+ }, d.prototype.unpack_int32 = function () {
+ const a = this.unpack_uint32();
+ return a < Math.pow(2, 31) ? a : a - Math.pow(2, 32);
+ }, d.prototype.unpack_int64 = function () {
+ const a = this.unpack_uint64();
+ return a < Math.pow(2, 63) ? a : a - Math.pow(2, 64);
+ }, d.prototype.unpack_raw = function (a) {
+ if (this.length < this.index + a) throw new Error(`BinaryPackFailure: index is out of range ${this.index} ${a} ${this.length}`);
+ const b = this.dataBuffer.slice(this.index, this.index + a);
+ return this.index += a, b;
+ }, d.prototype.unpack_string = function (a) {
+ for (var b, c, d = this.read(a), e = 0, f = ''; e < a;) b = d[e], b < 128 ? (f += String.fromCharCode(b), e++) : (192 ^ b) < 32 ? (c = (192 ^ b) << 6 | 63 & d[e + 1], f += String.fromCharCode(c), e += 2) : (c = (15 & b) << 12 | (63 & d[e + 1]) << 6 | 63 & d[e + 2], f += String.fromCharCode(c), e += 3);
+ return this.index += a, f;
+ }, d.prototype.unpack_array = function (a) {
+ for (var b = new Array(a), c = 0; c < a; c++) b[c] = this.unpack();
+ return b;
+ }, d.prototype.unpack_map = function (a) {
+ for (var b = {}, c = 0; c < a; c++) {
+ const d = this.unpack();
+ const
+ e = this.unpack();
+ b[d] = e;
+ }
+ return b;
+ }, d.prototype.unpack_float = function () {
+ const a = this.unpack_uint32();
+ const b = a >> 31;
+ const c = (a >> 23 & 255) - 127;
+ const
+ d = 8388607 & a | 8388608;
+ return (b == 0 ? 1 : -1) * d * Math.pow(2, c - 23);
+ }, d.prototype.unpack_double = function () {
+ const a = this.unpack_uint32();
+ const b = this.unpack_uint32();
+ const c = a >> 31;
+ const d = (a >> 20 & 2047) - 1023;
+
+
+ const e = 1048575 & a | 1048576;
+ const
+ f = e * Math.pow(2, d - 20) + b * Math.pow(2, d - 52);
+ return (c == 0 ? 1 : -1) * f;
+ }, d.prototype.read = function (a) {
+ const b = this.index;
+ if (b + a <= this.length) return this.dataView.subarray(b, b + a);
+ throw new Error('BinaryPackFailure: read index out of range');
+ }, e.prototype.getBuffer = function () {
+ return this.bufferBuilder.getBuffer();
+ }, e.prototype.pack = function (a) {
+ const b = typeof a;
+ if (b == 'string') {
+ this.pack_string(a);
+ } else if (b == 'number') {
+ Math.floor(a) === a ? this.pack_integer(a) : this.pack_double(a);
+ } else if (b == 'boolean') {
+ !0 === a ? this.bufferBuilder.append(195) : !1 === a && this.bufferBuilder.append(194);
+ } else if (b == 'undefined') {
+ this.bufferBuilder.append(192);
+ } else {
+ if (b != 'object') throw new Error(`Type "${b}" not yet supported`);
+ if (a === null) {
+ this.bufferBuilder.append(192);
+ } else {
+ const c = a.constructor;
+ if (c == Array) {
+ this.pack_array(a);
+ } else if (c == Blob || c == File) {
+ this.pack_bin(a);
+ } else if (c == ArrayBuffer) {
+ i.useArrayBufferView ? this.pack_bin(new Uint8Array(a)) : this.pack_bin(a);
+ } else if ('BYTES_PER_ELEMENT' in a) {
+ i.useArrayBufferView ? this.pack_bin(new Uint8Array(a.buffer)) : this.pack_bin(a.buffer);
+ } else if (c == Object) {
+ this.pack_object(a);
+ } else if (c == Date) {
+ this.pack_string(a.toString());
+ } else {
+ if (typeof a.toBinaryPack !== 'function') throw new Error(`Type "${c.toString()}" not yet supported`);
+ this.bufferBuilder.append(a.toBinaryPack());
+ }
+ }
+ }
+ this.bufferBuilder.flush();
+ }, e.prototype.pack_bin = function (a) {
+ const b = a.length || a.byteLength || a.size;
+ if (b <= 15) {
+ this.pack_uint8(160 + b);
+ } else if (b <= 65535) {
+ this.bufferBuilder.append(218), this.pack_uint16(b);
+ } else {
+ if (!(b <= 4294967295)) throw new Error('Invalid length');
+ this.bufferBuilder.append(219), this.pack_uint32(b);
+ }
+ this.bufferBuilder.append(a);
+ }, e.prototype.pack_string = function (a) {
+ const b = g(a);
+ if (b <= 15) {
+ this.pack_uint8(176 + b);
+ } else if (b <= 65535) {
+ this.bufferBuilder.append(216), this.pack_uint16(b);
+ } else {
+ if (!(b <= 4294967295)) throw new Error('Invalid length');
+ this.bufferBuilder.append(217), this.pack_uint32(b);
+ }
+ this.bufferBuilder.append(a);
+ }, e.prototype.pack_array = function (a) {
+ const b = a.length;
+ if (b <= 15) {
+ this.pack_uint8(144 + b);
+ } else if (b <= 65535) {
+ this.bufferBuilder.append(220), this.pack_uint16(b);
+ } else {
+ if (!(b <= 4294967295)) throw new Error('Invalid length');
+ this.bufferBuilder.append(221), this.pack_uint32(b);
+ }
+ for (let c = 0; c < b; c++) this.pack(a[c]);
+ }, e.prototype.pack_integer = function (a) {
+ if (a >= -32 && a <= 127) {
+ this.bufferBuilder.append(255 & a);
+ } else if (a >= 0 && a <= 255) {
+ this.bufferBuilder.append(204), this.pack_uint8(a);
+ } else if (a >= -128 && a <= 127) {
+ this.bufferBuilder.append(208), this.pack_int8(a);
+ } else if (a >= 0 && a <= 65535) {
+ this.bufferBuilder.append(205), this.pack_uint16(a);
+ } else if (a >= -32768 && a <= 32767) {
+ this.bufferBuilder.append(209), this.pack_int16(a);
+ } else if (a >= 0 && a <= 4294967295) {
+ this.bufferBuilder.append(206), this.pack_uint32(a);
+ } else if (a >= -2147483648 && a <= 2147483647) {
+ this.bufferBuilder.append(210), this.pack_int32(a);
+ } else if (a >= -0x8000000000000000 && a <= 0x8000000000000000) {
+ this.bufferBuilder.append(211), this.pack_int64(a);
+ } else {
+ if (!(a >= 0 && a <= 0x10000000000000000)) throw new Error('Invalid integer');
+ this.bufferBuilder.append(207), this.pack_uint64(a);
+ }
+ }, e.prototype.pack_double = function (a) {
+ let b = 0;
+ a < 0 && (b = 1, a = -a);
+ const c = Math.floor(Math.log(a) / Math.LN2);
+ const d = a / Math.pow(2, c) - 1;
+ const e = Math.floor(d * Math.pow(2, 52));
+
+
+ const f = Math.pow(2, 32);
+ const g = b << 31 | c + 1023 << 20 | e / f & 1048575;
+ const
+ h = e % f;
+ this.bufferBuilder.append(203), this.pack_int32(g), this.pack_int32(h);
+ }, e.prototype.pack_object = function (a) {
+ const b = Object.keys(a);
+ const
+ c = b.length;
+ if (c <= 15) {
+ this.pack_uint8(128 + c);
+ } else if (c <= 65535) {
+ this.bufferBuilder.append(222), this.pack_uint16(c);
+ } else {
+ if (!(c <= 4294967295)) throw new Error('Invalid length');
+ this.bufferBuilder.append(223), this.pack_uint32(c);
+ }
+ for (const d in a) a.hasOwnProperty(d) && (this.pack(d), this.pack(a[d]));
+ }, e.prototype.pack_uint8 = function (a) {
+ this.bufferBuilder.append(a);
+ }, e.prototype.pack_uint16 = function (a) {
+ this.bufferBuilder.append(a >> 8), this.bufferBuilder.append(255 & a);
+ }, e.prototype.pack_uint32 = function (a) {
+ const b = 4294967295 & a;
+ this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b);
+ }, e.prototype.pack_uint64 = function (a) {
+ const b = a / Math.pow(2, 32);
+ const
+ c = a % Math.pow(2, 32);
+ this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b), this.bufferBuilder.append((4278190080 & c) >>> 24), this.bufferBuilder.append((16711680 & c) >>> 16), this.bufferBuilder.append((65280 & c) >>> 8), this.bufferBuilder.append(255 & c);
+ }, e.prototype.pack_int8 = function (a) {
+ this.bufferBuilder.append(255 & a);
+ }, e.prototype.pack_int16 = function (a) {
+ this.bufferBuilder.append((65280 & a) >> 8), this.bufferBuilder.append(255 & a);
+ }, e.prototype.pack_int32 = function (a) {
+ this.bufferBuilder.append(a >>> 24 & 255), this.bufferBuilder.append((16711680 & a) >>> 16), this.bufferBuilder.append((65280 & a) >>> 8), this.bufferBuilder.append(255 & a);
+ }, e.prototype.pack_int64 = function (a) {
+ const b = Math.floor(a / Math.pow(2, 32));
+ const
+ c = a % Math.pow(2, 32);
+ this.bufferBuilder.append((4278190080 & b) >>> 24), this.bufferBuilder.append((16711680 & b) >>> 16), this.bufferBuilder.append((65280 & b) >>> 8), this.bufferBuilder.append(255 & b), this.bufferBuilder.append((4278190080 & c) >>> 24), this.bufferBuilder.append((16711680 & c) >>> 16), this.bufferBuilder.append((65280 & c) >>> 8), this.bufferBuilder.append(255 & c);
+ };
+ }, { './bufferbuilder': 11 }],
+ 11: [function (a, b, c) {
+ function d() {
+ this._pieces = [], this._parts = [];
+ }
+
+ const e = {};
+ e.useBlobBuilder = (function () {
+ try {
+ return new Blob([]), !1;
+ } catch (a) {
+ return !0;
+ }
+ }()), e.useArrayBufferView = !e.useBlobBuilder && (function () {
+ try {
+ return new Blob([new Uint8Array([])]).size === 0;
+ } catch (a) {
+ return !0;
+ }
+ }()), b.exports.binaryFeatures = e;
+ let f = b.exports.BlobBuilder;
+ typeof window !== 'undefined' && (f = b.exports.BlobBuilder = window.WebKitBlobBuilder || window.MozBlobBuilder || window.MSBlobBuilder || window.BlobBuilder), d.prototype.append = function (a) {
+ typeof a === 'number' ? this._pieces.push(a) : (this.flush(), this._parts.push(a));
+ }, d.prototype.flush = function () {
+ if (this._pieces.length > 0) {
+ let a = new Uint8Array(this._pieces);
+ e.useArrayBufferView || (a = a.buffer), this._parts.push(a), this._pieces = [];
+ }
+ }, d.prototype.getBuffer = function () {
+ if (this.flush(), e.useBlobBuilder) {
+ for (var a = new f(), b = 0, c = this._parts.length; b < c; b++) a.append(this._parts[b]);
+ return a.getBlob();
+ }
+ return new Blob(this._parts);
+ }, b.exports.BufferBuilder = d;
+ }, {}],
+ 12: [function (a, b, c) {
+ function d(a, b) {
+ if (!(this instanceof d)) return new d(a);
+ this._dc = a, e.debug = b, this._outgoing = {}, this._incoming = {}, this._received = {}, this._window = 1e3, this._mtu = 500, this._interval = 0, this._count = 0, this._queue = [], this._setupDC();
+ }
+
+ var e = a('./util');
+ d.prototype.send = function (a) {
+ const b = e.pack(a);
+ if (b.size < this._mtu) return void this._handleSend(['no', b]);
+ this._outgoing[this._count] = {
+ ack: 0,
+ chunks: this._chunk(b),
+ }, e.debug && (this._outgoing[this._count].timer = new Date()), this._sendWindowedChunks(this._count), this._count += 1;
+ }, d.prototype._setupInterval = function () {
+ const a = this;
+ this._timeout = setInterval(() => {
+ const b = a._queue.shift();
+ if (b._multiple) for (let c = 0, d = b.length; c < d; c += 1) a._intervalSend(b[c]); else a._intervalSend(b);
+ }, this._interval);
+ }, d.prototype._intervalSend = function (a) {
+ const b = this;
+ a = e.pack(a), e.blobToBinaryString(a, (a) => {
+ b._dc.send(a);
+ }), b._queue.length === 0 && (clearTimeout(b._timeout), b._timeout = null);
+ }, d.prototype._processAcks = function () {
+ for (const a in this._outgoing) this._outgoing.hasOwnProperty(a) && this._sendWindowedChunks(a);
+ }, d.prototype._handleSend = function (a) {
+ for (var b = !0, c = 0, d = this._queue.length; c < d; c += 1) {
+ const e = this._queue[c];
+ e === a ? b = !1 : e._multiple && e.indexOf(a) !== -1 && (b = !1);
+ }
+ b && (this._queue.push(a), this._timeout || this._setupInterval());
+ }, d.prototype._setupDC = function () {
+ const a = this;
+ this._dc.onmessage = function (b) {
+ let c = b.data;
+ if (c.constructor === String) {
+ const d = e.binaryStringToArrayBuffer(c);
+ c = e.unpack(d), a._handleMessage(c);
+ }
+ };
+ }, d.prototype._handleMessage = function (a) {
+ let b;
+ const c = a[1];
+ const d = this._incoming[c];
+ const
+ f = this._outgoing[c];
+ switch (a[0]) {
+ case 'no':
+ var g = c;
+ g && this.onmessage(e.unpack(g));
+ break;
+ case 'end':
+ if (b = d, this._received[c] = a[2], !b) break;
+ this._ack(c);
+ break;
+ case 'ack':
+ if (b = f) {
+ const h = a[2];
+ b.ack = Math.max(h, b.ack), b.ack >= b.chunks.length ? (e.log('Time: ', new Date() - b.timer), delete this._outgoing[c]) : this._processAcks();
+ }
+ break;
+ case 'chunk':
+ if (!(b = d)) {
+ if (!0 === this._received[c]) break;
+ b = {
+ ack: ['ack', c, 0],
+ chunks: []
+ }, this._incoming[c] = b;
+ }
+ var i = a[2];
+ var
+ j = a[3];
+ b.chunks[i] = new Uint8Array(j), i === b.ack[2] && this._calculateNextAck(c), this._ack(c);
+ break;
+ default:
+ this._handleSend(a);
+ }
+ }, d.prototype._chunk = function (a) {
+ for (var b = [], c = a.size, d = 0; d < c;) {
+ const f = Math.min(c, d + this._mtu);
+ const g = a.slice(d, f);
+ const
+ h = { payload: g };
+ b.push(h), d = f;
+ }
+ return e.log('Created', b.length, 'chunks.'), b;
+ }, d.prototype._ack = function (a) {
+ const b = this._incoming[a].ack;
+ this._received[a] === b[2] && (this._complete(a), this._received[a] = !0), this._handleSend(b);
+ }, d.prototype._calculateNextAck = function (a) {
+ for (var b = this._incoming[a], c = b.chunks, d = 0, e = c.length; d < e; d += 1) if (void 0 === c[d]) return void (b.ack[2] = d);
+ b.ack[2] = c.length;
+ }, d.prototype._sendWindowedChunks = function (a) {
+ e.log('sendWindowedChunks for: ', a);
+ for (var b = this._outgoing[a], c = b.chunks, d = [], f = Math.min(b.ack + this._window, c.length), g = b.ack; g < f; g += 1) c[g].sent && g !== b.ack || (c[g].sent = !0, d.push(['chunk', a, g, c[g].payload]));
+ b.ack + this._window >= c.length && d.push(['end', a, c.length]), d._multiple = !0, this._handleSend(d);
+ }, d.prototype._complete = function (a) {
+ e.log('Completed called for', a);
+ const b = this;
+ const c = this._incoming[a].chunks;
+ const
+ d = new Blob(c);
+ e.blobToArrayBuffer(d, (a) => {
+ b.onmessage(e.unpack(a));
+ }), delete this._incoming[a];
+ }, d.higherBandwidthSDP = function (a) {
+ let b = navigator.appVersion.match(/Chrome\/(.*?) /);
+ if (b && (b = parseInt(b[1].split('.')
+ .shift())) < 31) {
+ const c = a.split('b=AS:30');
+ if (c.length > 1) return `${c[0]}b=AS:102400${c[1]}`;
+ }
+ return a;
+ }, d.prototype.onmessage = function (a) {
+ }, b.exports.Reliable = d;
+ }, { './util': 13 }],
+ 13: [function (a, b, c) {
+ const d = a('js-binarypack');
+ var
+ e = {
+ debug: !1,
+ inherits(a, b) {
+ a.super_ = b, a.prototype = Object.create(b.prototype, {
+ constructor: {
+ value: a,
+ enumerable: !1,
+ writable: !0,
+ configurable: !0,
+ },
+ });
+ },
+ extend(a, b) {
+ for (const c in b) b.hasOwnProperty(c) && (a[c] = b[c]);
+ return a;
+ },
+ pack: d.pack,
+ unpack: d.unpack,
+ log() {
+ if (e.debug) {
+ for (var a = [], b = 0; b < arguments.length; b++) a[b] = arguments[b];
+ a.unshift('Reliable: '), console.log(...a);
+ }
+ },
+ setZeroTimeout: (function (a) {
+ function b(b) {
+ d.push(b), a.postMessage(e, '*');
+ }
+
+ function c(b) {
+ b.source == a && b.data == e && (b.stopPropagation && b.stopPropagation(), d.length && d.shift()());
+ }
+
+ var d = [];
+ var
+ e = 'zero-timeout-message';
+ return a.addEventListener ? a.addEventListener('message', c, !0) : a.attachEvent && a.attachEvent('onmessage', c), b;
+ }(this)),
+ blobToArrayBuffer(a, b) {
+ const c = new FileReader();
+ c.onload = function (a) {
+ b(a.target.result);
+ }, c.readAsArrayBuffer(a);
+ },
+ blobToBinaryString(a, b) {
+ const c = new FileReader();
+ c.onload = function (a) {
+ b(a.target.result);
+ }, c.readAsBinaryString(a);
+ },
+ binaryStringToArrayBuffer(a) {
+ for (var b = new Uint8Array(a.length), c = 0; c < a.length; c++) b[c] = 255 & a.charCodeAt(c);
+ return b.buffer;
+ },
+ randomToken() {
+ return Math.random()
+ .toString(36)
+ .substr(2);
+ },
+ };
+ b.exports = e;
+ }, { 'js-binarypack': 10 }],
+}, {}, [3]));
diff --git a/public/scripts/wordlist.js b/public/scripts/wordlist.js
index bce5fca..4360437 100644
--- a/public/scripts/wordlist.js
+++ b/public/scripts/wordlist.js
@@ -1,1635 +1,1642 @@
+/*
+ * wordlist.js
+ * Copyright (c) 2019, Texx
+ * License: MIT
+ * See https://github.com/texxme/Texx/blob/master/LICENSE
+ */
+
module.exports = [
- "academy",
- "acrobat",
- "active",
- "actor",
- "adam",
- "admiral",
- "adrian",
- "africa",
- "agenda",
- "agent",
- "airline",
- "airport",
- "aladdin",
- "alarm",
- "alaska",
- "albert",
- "albino",
- "album",
- "alcohol",
- "alex",
- "algebra",
- "alibi",
- "alice",
- "alien",
- "alpha",
- "alpine",
- "amadeus",
- "amanda",
- "amazon",
- "amber",
- "america",
- "amigo",
- "analog",
- "anatomy",
- "angel",
- "animal",
- "antenna",
- "antonio",
- "apollo",
- "april",
- "archive",
- "arctic",
- "arizona",
- "arnold",
- "aroma",
- "arthur",
- "artist",
- "asia",
- "aspect",
- "aspirin",
- "athena",
- "athlete",
- "atlas",
- "audio",
- "august",
- "austria",
- "axiom",
- "aztec",
- "balance",
- "ballad",
- "banana",
- "bandit",
- "banjo",
- "barcode",
- "baron",
- "basic",
- "battery",
- "belgium",
- "berlin",
- "bermuda",
- "bernard",
- "bikini",
- "binary",
- "bingo",
- "biology",
- "block",
- "blonde",
- "bonus",
- "boris",
- "boston",
- "boxer",
- "brandy",
- "bravo",
- "brazil",
- "bronze",
- "brown",
- "bruce",
- "bruno",
- "burger",
- "burma",
- "cabinet",
- "cactus",
- "cafe",
- "cairo",
- "cake",
- "calypso",
- "camel",
- "camera",
- "campus",
- "canada",
- "canal",
- "cannon",
- "canoe",
- "cantina",
- "canvas",
- "canyon",
- "capital",
- "caramel",
- "caravan",
- "carbon",
- "cargo",
- "carlo",
- "carol",
- "carpet",
- "cartel",
- "casino",
- "castle",
- "castro",
- "catalog",
- "caviar",
- "cecilia",
- "cement",
- "center",
- "century",
- "ceramic",
- "chamber",
- "chance",
- "change",
- "chaos",
- "charlie",
- "charm",
- "charter",
- "chef",
- "chemist",
- "cherry",
- "chess",
- "chicago",
- "chicken",
- "chief",
- "china",
- "cigar",
- "cinema",
- "circus",
- "citizen",
- "city",
- "clara",
- "classic",
- "claudia",
- "clean",
- "client",
- "climax",
- "clinic",
- "clock",
- "club",
- "cobra",
- "coconut",
- "cola",
- "collect",
- "colombo",
- "colony",
- "color",
- "combat",
- "comedy",
- "comet",
- "command",
- "compact",
- "company",
- "complex",
- "concept",
- "concert",
- "connect",
- "consul",
- "contact",
- "context",
- "contour",
- "control",
- "convert",
- "copy",
- "corner",
- "corona",
- "correct",
- "cosmos",
- "couple",
- "courage",
- "cowboy",
- "craft",
- "crash",
- "credit",
- "cricket",
- "critic",
- "crown",
- "crystal",
- "cuba",
- "culture",
- "dallas",
- "dance",
- "daniel",
- "david",
- "decade",
- "decimal",
- "deliver",
- "delta",
- "deluxe",
- "demand",
- "demo",
- "denmark",
- "derby",
- "design",
- "detect",
- "develop",
- "diagram",
- "dialog",
- "diamond",
- "diana",
- "diego",
- "diesel",
- "diet",
- "digital",
- "dilemma",
- "diploma",
- "direct",
- "disco",
- "disney",
- "distant",
- "doctor",
- "dollar",
- "dominic",
- "domino",
- "donald",
- "dragon",
- "drama",
- "dublin",
- "duet",
- "dynamic",
- "east",
- "ecology",
- "economy",
- "edgar",
- "egypt",
- "elastic",
- "elegant",
- "element",
- "elite",
- "elvis",
- "email",
- "energy",
- "engine",
- "english",
- "episode",
- "equator",
- "escort",
- "ethnic",
- "europe",
- "everest",
- "evident",
- "exact",
- "example",
- "exit",
- "exotic",
- "export",
- "express",
- "extra",
- "fabric",
- "factor",
- "falcon",
- "family",
- "fantasy",
- "fashion",
- "fiber",
- "fiction",
- "fidel",
- "fiesta",
- "figure",
- "film",
- "filter",
- "final",
- "finance",
- "finish",
- "finland",
- "flash",
- "florida",
- "flower",
- "fluid",
- "flute",
- "focus",
- "ford",
- "forest",
- "formal",
- "format",
- "formula",
- "fortune",
- "forum",
- "fragile",
- "france",
- "frank",
- "friend",
- "frozen",
- "future",
- "gabriel",
- "galaxy",
- "gallery",
- "gamma",
- "garage",
- "garden",
- "garlic",
- "gemini",
- "general",
- "genetic",
- "genius",
- "germany",
- "global",
- "gloria",
- "golf",
- "gondola",
- "gong",
- "good",
- "gordon",
- "gorilla",
- "grand",
- "granite",
- "graph",
- "green",
- "group",
- "guide",
- "guitar",
- "guru",
- "hand",
- "happy",
- "harbor",
- "harmony",
- "harvard",
- "havana",
- "hawaii",
- "helena",
- "hello",
- "henry",
- "hilton",
- "history",
- "horizon",
- "hotel",
- "human",
- "humor",
- "icon",
- "idea",
- "igloo",
- "igor",
- "image",
- "impact",
- "import",
- "index",
- "india",
- "indigo",
- "input",
- "insect",
- "instant",
- "iris",
- "italian",
- "jacket",
- "jacob",
- "jaguar",
- "janet",
- "japan",
- "jargon",
- "jazz",
- "jeep",
- "john",
- "joker",
- "jordan",
- "jumbo",
- "june",
- "jungle",
- "junior",
- "jupiter",
- "karate",
- "karma",
- "kayak",
- "kermit",
- "kilo",
- "king",
- "koala",
- "korea",
- "labor",
- "lady",
- "lagoon",
- "laptop",
- "laser",
- "latin",
- "lava",
- "lecture",
- "left",
- "legal",
- "lemon",
- "level",
- "lexicon",
- "liberal",
- "libra",
- "limbo",
- "limit",
- "linda",
- "linear",
- "lion",
- "liquid",
- "liter",
- "little",
- "llama",
- "lobby",
- "lobster",
- "local",
- "logic",
- "logo",
- "lola",
- "london",
- "lotus",
- "lucas",
- "lunar",
- "machine",
- "macro",
- "madam",
- "madonna",
- "madrid",
- "maestro",
- "magic",
- "magnet",
- "magnum",
- "major",
- "mama",
- "mambo",
- "manager",
- "mango",
- "manila",
- "marco",
- "marina",
- "market",
- "mars",
- "martin",
- "marvin",
- "master",
- "matrix",
- "maximum",
- "media",
- "medical",
- "mega",
- "melody",
- "melon",
- "memo",
- "mental",
- "mentor",
- "menu",
- "mercury",
- "message",
- "metal",
- "meteor",
- "meter",
- "method",
- "metro",
- "mexico",
- "miami",
- "micro",
- "million",
- "mineral",
- "minimum",
- "minus",
- "minute",
- "miracle",
- "mirage",
- "miranda",
- "mister",
- "mixer",
- "mobile",
- "model",
- "modem",
- "modern",
- "modular",
- "moment",
- "monaco",
- "monica",
- "monitor",
- "mono",
- "monster",
- "montana",
- "morgan",
- "motel",
- "motif",
- "motor",
- "mozart",
- "multi",
- "museum",
- "music",
- "mustang",
- "natural",
- "neon",
- "nepal",
- "neptune",
- "nerve",
- "neutral",
- "nevada",
- "news",
- "ninja",
- "nirvana",
- "normal",
- "nova",
- "novel",
- "nuclear",
- "numeric",
- "nylon",
- "oasis",
- "object",
- "observe",
- "ocean",
- "octopus",
- "olivia",
- "olympic",
- "omega",
- "opera",
- "optic",
- "optimal",
- "orange",
- "orbit",
- "organic",
- "orient",
- "origin",
- "orlando",
- "oscar",
- "oxford",
- "oxygen",
- "ozone",
- "pablo",
- "pacific",
- "pagoda",
- "palace",
- "pamela",
- "panama",
- "panda",
- "panel",
- "panic",
- "paradox",
- "pardon",
- "paris",
- "parker",
- "parking",
- "parody",
- "partner",
- "passage",
- "passive",
- "pasta",
- "pastel",
- "patent",
- "patriot",
- "patrol",
- "patron",
- "pegasus",
- "pelican",
- "penguin",
- "pepper",
- "percent",
- "perfect",
- "perfume",
- "period",
- "permit",
- "person",
- "peru",
- "phone",
- "photo",
- "piano",
- "picasso",
- "picnic",
- "picture",
- "pigment",
- "pilgrim",
- "pilot",
- "pirate",
- "pixel",
- "pizza",
- "planet",
- "plasma",
- "plaster",
- "plastic",
- "plaza",
- "pocket",
- "poem",
- "poetic",
- "poker",
- "polaris",
- "police",
- "politic",
- "polo",
- "polygon",
- "pony",
- "popcorn",
- "popular",
- "postage",
- "postal",
- "precise",
- "prefix",
- "premium",
- "present",
- "price",
- "prince",
- "printer",
- "prism",
- "private",
- "product",
- "profile",
- "program",
- "project",
- "protect",
- "proton",
- "public",
- "pulse",
- "puma",
- "pyramid",
- "queen",
- "radar",
- "radio",
- "random",
- "rapid",
- "rebel",
- "record",
- "recycle",
- "reflex",
- "reform",
- "regard",
- "regular",
- "relax",
- "report",
- "reptile",
- "reverse",
- "ricardo",
- "ringo",
- "ritual",
- "robert",
- "robot",
- "rocket",
- "rodeo",
- "romeo",
- "royal",
- "russian",
- "safari",
- "salad",
- "salami",
- "salmon",
- "salon",
- "salute",
- "samba",
- "sandra",
- "santana",
- "sardine",
- "school",
- "screen",
- "script",
- "second",
- "secret",
- "section",
- "segment",
- "select",
- "seminar",
- "senator",
- "senior",
- "sensor",
- "serial",
- "service",
- "sheriff",
- "shock",
- "sierra",
- "signal",
- "silicon",
- "silver",
- "similar",
- "simon",
- "single",
- "siren",
- "slogan",
- "social",
- "soda",
- "solar",
- "solid",
- "solo",
- "sonic",
- "soviet",
- "special",
- "speed",
- "spiral",
- "spirit",
- "sport",
- "static",
- "station",
- "status",
- "stereo",
- "stone",
- "stop",
- "street",
- "strong",
- "student",
- "studio",
- "style",
- "subject",
- "sultan",
- "super",
- "susan",
- "sushi",
- "suzuki",
- "switch",
- "symbol",
- "system",
- "tactic",
- "tahiti",
- "talent",
- "tango",
- "tarzan",
- "taxi",
- "telex",
- "tempo",
- "tennis",
- "texas",
- "textile",
- "theory",
- "thermos",
- "tiger",
- "titanic",
- "tokyo",
- "tomato",
- "topic",
- "tornado",
- "toronto",
- "torpedo",
- "total",
- "totem",
- "tourist",
- "tractor",
- "traffic",
- "transit",
- "trapeze",
- "travel",
- "tribal",
- "trick",
- "trident",
- "trilogy",
- "tripod",
- "tropic",
- "trumpet",
- "tulip",
- "tuna",
- "turbo",
- "twist",
- "ultra",
- "uniform",
- "union",
- "uranium",
- "vacuum",
- "valid",
- "vampire",
- "vanilla",
- "vatican",
- "velvet",
- "ventura",
- "venus",
- "vertigo",
- "veteran",
- "victor",
- "video",
- "vienna",
- "viking",
- "village",
- "vincent",
- "violet",
- "violin",
- "virtual",
- "virus",
- "visa",
- "vision",
- "visitor",
- "visual",
- "vitamin",
- "viva",
- "vocal",
- "vodka",
- "volcano",
- "voltage",
- "volume",
- "voyage",
- "water",
- "weekend",
- "welcome",
- "western",
- "window",
- "winter",
- "wizard",
- "wolf",
- "world",
- "xray",
- "yankee",
- "yoga",
- "yogurt",
- "yoyo",
- "zebra",
- "zero",
- "zigzag",
- "zipper",
- "zodiac",
- "zoom",
- "abraham",
- "action",
- "address",
- "alabama",
- "alfred",
- "almond",
- "ammonia",
- "analyze",
- "annual",
- "answer",
- "apple",
- "arena",
- "armada",
- "arsenal",
- "atlanta",
- "atomic",
- "avenue",
- "average",
- "bagel",
- "baker",
- "ballet",
- "bambino",
- "bamboo",
- "barbara",
- "basket",
- "bazaar",
- "benefit",
- "bicycle",
- "bishop",
- "blitz",
- "bonjour",
- "bottle",
- "bridge",
- "british",
- "brother",
- "brush",
- "budget",
- "cabaret",
- "cadet",
- "candle",
- "capitan",
- "capsule",
- "career",
- "cartoon",
- "channel",
- "chapter",
- "cheese",
- "circle",
- "cobalt",
- "cockpit",
- "college",
- "compass",
- "comrade",
- "condor",
- "crimson",
- "cyclone",
- "darwin",
- "declare",
- "degree",
- "delete",
- "delphi",
- "denver",
- "desert",
- "divide",
- "dolby",
- "domain",
- "domingo",
- "double",
- "drink",
- "driver",
- "eagle",
- "earth",
- "echo",
- "eclipse",
- "editor",
- "educate",
- "edward",
- "effect",
- "electra",
- "emerald",
- "emotion",
- "empire",
- "empty",
- "escape",
- "eternal",
- "evening",
- "exhibit",
- "expand",
- "explore",
- "extreme",
- "ferrari",
- "first",
- "flag",
- "folio",
- "forget",
- "forward",
- "freedom",
- "fresh",
- "friday",
- "fuji",
- "galileo",
- "garcia",
- "genesis",
- "gold",
- "gravity",
- "habitat",
- "hamlet",
- "harlem",
- "helium",
- "holiday",
- "house",
- "hunter",
- "ibiza",
- "iceberg",
- "imagine",
- "infant",
- "isotope",
- "jackson",
- "jamaica",
- "jasmine",
- "java",
- "jessica",
- "judo",
- "kitchen",
- "lazarus",
- "letter",
- "license",
- "lithium",
- "loyal",
- "lucky",
- "magenta",
- "mailbox",
- "manual",
- "marble",
- "mary",
- "maxwell",
- "mayor",
- "milk",
- "monarch",
- "monday",
- "money",
- "morning",
- "mother",
- "mystery",
- "native",
- "nectar",
- "nelson",
- "network",
- "next",
- "nikita",
- "nobel",
- "nobody",
- "nominal",
- "norway",
- "nothing",
- "number",
- "october",
- "office",
- "oliver",
- "opinion",
- "option",
- "order",
- "outside",
- "package",
- "pancake",
- "pandora",
- "panther",
- "papa",
- "patient",
- "pattern",
- "pedro",
- "pencil",
- "people",
- "phantom",
- "philips",
- "pioneer",
- "pluto",
- "podium",
- "portal",
- "potato",
- "prize",
- "process",
- "protein",
- "proxy",
- "pump",
- "pupil",
- "python",
- "quality",
- "quarter",
- "quiet",
- "rabbit",
- "radical",
- "radius",
- "rainbow",
- "ralph",
- "ramirez",
- "ravioli",
- "raymond",
- "respect",
- "respond",
- "result",
- "resume",
- "retro",
- "richard",
- "right",
- "risk",
- "river",
- "roger",
- "roman",
- "rondo",
- "sabrina",
- "salary",
- "salsa",
- "sample",
- "samuel",
- "saturn",
- "savage",
- "scarlet",
- "scoop",
- "scorpio",
- "scratch",
- "scroll",
- "sector",
- "serpent",
- "shadow",
- "shampoo",
- "sharon",
- "sharp",
- "short",
- "shrink",
- "silence",
- "silk",
- "simple",
- "slang",
- "smart",
- "smoke",
- "snake",
- "society",
- "sonar",
- "sonata",
- "soprano",
- "source",
- "sparta",
- "sphere",
- "spider",
- "sponsor",
- "spring",
- "acid",
- "adios",
- "agatha",
- "alamo",
- "alert",
- "almanac",
- "aloha",
- "andrea",
- "anita",
- "arcade",
- "aurora",
- "avalon",
- "baby",
- "baggage",
- "balloon",
- "bank",
- "basil",
- "begin",
- "biscuit",
- "blue",
- "bombay",
- "brain",
- "brenda",
- "brigade",
- "cable",
- "carmen",
- "cello",
- "celtic",
- "chariot",
- "chrome",
- "citrus",
- "civil",
- "cloud",
- "common",
- "compare",
- "cool",
- "copper",
- "coral",
- "crater",
- "cubic",
- "cupid",
- "cycle",
- "depend",
- "door",
- "dream",
- "dynasty",
- "edison",
- "edition",
- "enigma",
- "equal",
- "eric",
- "event",
- "evita",
- "exodus",
- "extend",
- "famous",
- "farmer",
- "food",
- "fossil",
- "frog",
- "fruit",
- "geneva",
- "gentle",
- "george",
- "giant",
- "gilbert",
- "gossip",
- "gram",
- "greek",
- "grille",
- "hammer",
- "harvest",
- "hazard",
- "heaven",
- "herbert",
- "heroic",
- "hexagon",
- "husband",
- "immune",
- "inca",
- "inch",
- "initial",
- "isabel",
- "ivory",
- "jason",
- "jerome",
- "joel",
- "joshua",
- "journal",
- "judge",
- "juliet",
- "jump",
- "justice",
- "kimono",
- "kinetic",
- "leonid",
- "lima",
- "maze",
- "medusa",
- "member",
- "memphis",
- "michael",
- "miguel",
- "milan",
- "mile",
- "miller",
- "mimic",
- "mimosa",
- "mission",
- "monkey",
- "moral",
- "moses",
- "mouse",
- "nancy",
- "natasha",
- "nebula",
- "nickel",
- "nina",
- "noise",
- "orchid",
- "oregano",
- "origami",
- "orinoco",
- "orion",
- "othello",
- "paper",
- "paprika",
- "prelude",
- "prepare",
- "pretend",
- "profit",
- "promise",
- "provide",
- "puzzle",
- "remote",
- "repair",
- "reply",
- "rival",
- "riviera",
- "robin",
- "rose",
- "rover",
- "rudolf",
- "saga",
- "sahara",
- "scholar",
- "shelter",
- "ship",
- "shoe",
- "sigma",
- "sister",
- "sleep",
- "smile",
- "spain",
- "spark",
- "split",
- "spray",
- "square",
- "stadium",
- "star",
- "storm",
- "story",
- "strange",
- "stretch",
- "stuart",
- "subway",
- "sugar",
- "sulfur",
- "summer",
- "survive",
- "sweet",
- "swim",
- "table",
- "taboo",
- "target",
- "teacher",
- "telecom",
- "temple",
- "tibet",
- "ticket",
- "tina",
- "today",
- "toga",
- "tommy",
- "tower",
- "trivial",
- "tunnel",
- "turtle",
- "twin",
- "uncle",
- "unicorn",
- "unique",
- "update",
- "valery",
- "vega",
- "version",
- "voodoo",
- "warning",
- "william",
- "wonder",
- "year",
- "yellow",
- "young",
- "absent",
- "absorb",
- "accent",
- "alfonso",
- "alias",
- "ambient",
- "andy",
- "anvil",
- "appear",
- "apropos",
- "archer",
- "ariel",
- "armor",
- "arrow",
- "austin",
- "avatar",
- "axis",
- "baboon",
- "bahama",
- "bali",
- "balsa",
- "bazooka",
- "beach",
- "beast",
- "beatles",
- "beauty",
- "before",
- "benny",
- "betty",
- "between",
- "beyond",
- "billy",
- "bison",
- "blast",
- "bless",
- "bogart",
- "bonanza",
- "book",
- "border",
- "brave",
- "bread",
- "break",
- "broken",
- "bucket",
- "buenos",
- "buffalo",
- "bundle",
- "button",
- "buzzer",
- "byte",
- "caesar",
- "camilla",
- "canary",
- "candid",
- "carrot",
- "cave",
- "chant",
- "child",
- "choice",
- "chris",
- "cipher",
- "clarion",
- "clark",
- "clever",
- "cliff",
- "clone",
- "conan",
- "conduct",
- "congo",
- "content",
- "costume",
- "cotton",
- "cover",
- "crack",
- "current",
- "danube",
- "data",
- "decide",
- "desire",
- "detail",
- "dexter",
- "dinner",
- "dispute",
- "donor",
- "druid",
- "drum",
- "easy",
- "eddie",
- "enjoy",
- "enrico",
- "epoxy",
- "erosion",
- "except",
- "exile",
- "explain",
- "fame",
- "fast",
- "father",
- "felix",
- "field",
- "fiona",
- "fire",
- "fish",
- "flame",
- "flex",
- "flipper",
- "float",
- "flood",
- "floor",
- "forbid",
- "forever",
- "fractal",
- "frame",
- "freddie",
- "front",
- "fuel",
- "gallop",
- "game",
- "garbo",
- "gate",
- "gibson",
- "ginger",
- "giraffe",
- "gizmo",
- "glass",
- "goblin",
- "gopher",
- "grace",
- "gray",
- "gregory",
- "grid",
- "griffin",
- "ground",
- "guest",
- "gustav",
- "gyro",
- "hair",
- "halt",
- "harris",
- "heart",
- "heavy",
- "herman",
- "hippie",
- "hobby",
- "honey",
- "hope",
- "horse",
- "hostel",
- "hydro",
- "imitate",
- "info",
- "ingrid",
- "inside",
- "invent",
- "invest",
- "invite",
- "iron",
- "ivan",
- "james",
- "jester",
- "jimmy",
- "join",
- "joseph",
- "juice",
- "julius",
- "july",
- "justin",
- "kansas",
- "karl",
- "kevin",
- "kiwi",
- "ladder",
- "lake",
- "laura",
- "learn",
- "legacy",
- "legend",
- "lesson",
- "life",
- "light",
- "list",
- "locate",
- "lopez",
- "lorenzo",
- "love",
- "lunch",
- "malta",
- "mammal",
- "margo",
- "marion",
- "mask",
- "match",
- "mayday",
- "meaning",
- "mercy",
- "middle",
- "mike",
- "mirror",
- "modest",
- "morph",
- "morris",
- "nadia",
- "nato",
- "navy",
- "needle",
- "neuron",
- "never",
- "newton",
- "nice",
- "night",
- "nissan",
- "nitro",
- "nixon",
- "north",
- "oberon",
- "octavia",
- "ohio",
- "olga",
- "open",
- "opus",
- "orca",
- "oval",
- "owner",
- "page",
- "paint",
- "palma",
- "parade",
- "parent",
- "parole",
- "paul",
- "peace",
- "pearl",
- "perform",
- "phoenix",
- "phrase",
- "pierre",
- "pinball",
- "place",
- "plate",
- "plato",
- "plume",
- "pogo",
- "point",
- "polite",
- "polka",
- "poncho",
- "powder",
- "prague",
- "press",
- "presto",
- "pretty",
- "prime",
- "promo",
- "quasi",
- "quest",
- "quick",
- "quiz",
- "quota",
- "race",
- "rachel",
- "raja",
- "ranger",
- "region",
- "remark",
- "rent",
- "reward",
- "rhino",
- "ribbon",
- "rider",
- "road",
- "rodent",
- "round",
- "rubber",
- "ruby",
- "rufus",
- "sabine",
- "saddle",
- "sailor",
- "saint",
- "salt",
- "satire",
- "scale",
- "scuba",
- "season",
- "secure",
- "shake",
- "shallow",
- "shannon",
- "shave",
- "shelf",
- "sherman",
- "shine",
- "shirt",
- "side",
- "sinatra",
- "sincere",
- "size",
- "slalom",
- "slow",
- "small",
- "snow",
- "sofia",
- "song",
- "sound",
- "south",
- "speech",
- "spell",
- "spend",
- "spoon",
- "stage",
- "stamp",
- "stand",
- "state",
- "stella",
- "stick",
- "sting",
- "stock",
- "store",
- "sunday",
- "sunset",
- "support",
- "sweden",
- "swing",
- "tape",
- "think",
- "thomas",
- "tictac",
- "time",
- "toast",
- "tobacco",
- "tonight",
- "torch",
- "torso",
- "touch",
- "toyota",
- "trade",
- "tribune",
- "trinity",
- "triton",
- "truck",
- "trust",
- "type",
- "under",
- "unit",
- "urban",
- "urgent",
- "user",
- "value",
- "vendor",
- "venice",
- "verona",
- "vibrate",
- "virgo",
- "visible",
- "vista",
- "vital",
- "voice",
- "vortex",
- "waiter",
- "watch",
- "wave",
- "weather",
- "wedding",
- "wheel",
- "whiskey",
- "wisdom",
- "deal",
- "null",
- "nurse",
- "quebec",
- "reserve",
- "reunion",
- "roof",
- "singer",
- "verbal",
- "amen",
- "ego",
- "fax",
- "jet",
- "job",
- "rio",
- "ski",
- "yes"
+ 'academy',
+ 'acrobat',
+ 'active',
+ 'actor',
+ 'adam',
+ 'admiral',
+ 'adrian',
+ 'africa',
+ 'agenda',
+ 'agent',
+ 'airline',
+ 'airport',
+ 'aladdin',
+ 'alarm',
+ 'alaska',
+ 'albert',
+ 'albino',
+ 'album',
+ 'alcohol',
+ 'alex',
+ 'algebra',
+ 'alibi',
+ 'alice',
+ 'alien',
+ 'alpha',
+ 'alpine',
+ 'amadeus',
+ 'amanda',
+ 'amazon',
+ 'amber',
+ 'america',
+ 'amigo',
+ 'analog',
+ 'anatomy',
+ 'angel',
+ 'animal',
+ 'antenna',
+ 'antonio',
+ 'apollo',
+ 'april',
+ 'archive',
+ 'arctic',
+ 'arizona',
+ 'arnold',
+ 'aroma',
+ 'arthur',
+ 'artist',
+ 'asia',
+ 'aspect',
+ 'aspirin',
+ 'athena',
+ 'athlete',
+ 'atlas',
+ 'audio',
+ 'august',
+ 'austria',
+ 'axiom',
+ 'aztec',
+ 'balance',
+ 'ballad',
+ 'banana',
+ 'bandit',
+ 'banjo',
+ 'barcode',
+ 'baron',
+ 'basic',
+ 'battery',
+ 'belgium',
+ 'berlin',
+ 'bermuda',
+ 'bernard',
+ 'bikini',
+ 'binary',
+ 'bingo',
+ 'biology',
+ 'block',
+ 'blonde',
+ 'bonus',
+ 'boris',
+ 'boston',
+ 'boxer',
+ 'brandy',
+ 'bravo',
+ 'brazil',
+ 'bronze',
+ 'brown',
+ 'bruce',
+ 'bruno',
+ 'burger',
+ 'burma',
+ 'cabinet',
+ 'cactus',
+ 'cafe',
+ 'cairo',
+ 'cake',
+ 'calypso',
+ 'camel',
+ 'camera',
+ 'campus',
+ 'canada',
+ 'canal',
+ 'cannon',
+ 'canoe',
+ 'cantina',
+ 'canvas',
+ 'canyon',
+ 'capital',
+ 'caramel',
+ 'caravan',
+ 'carbon',
+ 'cargo',
+ 'carlo',
+ 'carol',
+ 'carpet',
+ 'cartel',
+ 'casino',
+ 'castle',
+ 'castro',
+ 'catalog',
+ 'caviar',
+ 'cecilia',
+ 'cement',
+ 'center',
+ 'century',
+ 'ceramic',
+ 'chamber',
+ 'chance',
+ 'change',
+ 'chaos',
+ 'charlie',
+ 'charm',
+ 'charter',
+ 'chef',
+ 'chemist',
+ 'cherry',
+ 'chess',
+ 'chicago',
+ 'chicken',
+ 'chief',
+ 'china',
+ 'cigar',
+ 'cinema',
+ 'circus',
+ 'citizen',
+ 'city',
+ 'clara',
+ 'classic',
+ 'claudia',
+ 'clean',
+ 'client',
+ 'climax',
+ 'clinic',
+ 'clock',
+ 'club',
+ 'cobra',
+ 'coconut',
+ 'cola',
+ 'collect',
+ 'colombo',
+ 'colony',
+ 'color',
+ 'combat',
+ 'comedy',
+ 'comet',
+ 'command',
+ 'compact',
+ 'company',
+ 'complex',
+ 'concept',
+ 'concert',
+ 'connect',
+ 'consul',
+ 'contact',
+ 'context',
+ 'contour',
+ 'control',
+ 'convert',
+ 'copy',
+ 'corner',
+ 'corona',
+ 'correct',
+ 'cosmos',
+ 'couple',
+ 'courage',
+ 'cowboy',
+ 'craft',
+ 'crash',
+ 'credit',
+ 'cricket',
+ 'critic',
+ 'crown',
+ 'crystal',
+ 'cuba',
+ 'culture',
+ 'dallas',
+ 'dance',
+ 'daniel',
+ 'david',
+ 'decade',
+ 'decimal',
+ 'deliver',
+ 'delta',
+ 'deluxe',
+ 'demand',
+ 'demo',
+ 'denmark',
+ 'derby',
+ 'design',
+ 'detect',
+ 'develop',
+ 'diagram',
+ 'dialog',
+ 'diamond',
+ 'diana',
+ 'diego',
+ 'diesel',
+ 'diet',
+ 'digital',
+ 'dilemma',
+ 'diploma',
+ 'direct',
+ 'disco',
+ 'disney',
+ 'distant',
+ 'doctor',
+ 'dollar',
+ 'dominic',
+ 'domino',
+ 'donald',
+ 'dragon',
+ 'drama',
+ 'dublin',
+ 'duet',
+ 'dynamic',
+ 'east',
+ 'ecology',
+ 'economy',
+ 'edgar',
+ 'egypt',
+ 'elastic',
+ 'elegant',
+ 'element',
+ 'elite',
+ 'elvis',
+ 'email',
+ 'energy',
+ 'engine',
+ 'english',
+ 'episode',
+ 'equator',
+ 'escort',
+ 'ethnic',
+ 'europe',
+ 'everest',
+ 'evident',
+ 'exact',
+ 'example',
+ 'exit',
+ 'exotic',
+ 'export',
+ 'express',
+ 'extra',
+ 'fabric',
+ 'factor',
+ 'falcon',
+ 'family',
+ 'fantasy',
+ 'fashion',
+ 'fiber',
+ 'fiction',
+ 'fidel',
+ 'fiesta',
+ 'figure',
+ 'film',
+ 'filter',
+ 'final',
+ 'finance',
+ 'finish',
+ 'finland',
+ 'flash',
+ 'florida',
+ 'flower',
+ 'fluid',
+ 'flute',
+ 'focus',
+ 'ford',
+ 'forest',
+ 'formal',
+ 'format',
+ 'formula',
+ 'fortune',
+ 'forum',
+ 'fragile',
+ 'france',
+ 'frank',
+ 'friend',
+ 'frozen',
+ 'future',
+ 'gabriel',
+ 'galaxy',
+ 'gallery',
+ 'gamma',
+ 'garage',
+ 'garden',
+ 'garlic',
+ 'gemini',
+ 'general',
+ 'genetic',
+ 'genius',
+ 'germany',
+ 'global',
+ 'gloria',
+ 'golf',
+ 'gondola',
+ 'gong',
+ 'good',
+ 'gordon',
+ 'gorilla',
+ 'grand',
+ 'granite',
+ 'graph',
+ 'green',
+ 'group',
+ 'guide',
+ 'guitar',
+ 'guru',
+ 'hand',
+ 'happy',
+ 'harbor',
+ 'harmony',
+ 'harvard',
+ 'havana',
+ 'hawaii',
+ 'helena',
+ 'hello',
+ 'henry',
+ 'hilton',
+ 'history',
+ 'horizon',
+ 'hotel',
+ 'human',
+ 'humor',
+ 'icon',
+ 'idea',
+ 'igloo',
+ 'igor',
+ 'image',
+ 'impact',
+ 'import',
+ 'index',
+ 'india',
+ 'indigo',
+ 'input',
+ 'insect',
+ 'instant',
+ 'iris',
+ 'italian',
+ 'jacket',
+ 'jacob',
+ 'jaguar',
+ 'janet',
+ 'japan',
+ 'jargon',
+ 'jazz',
+ 'jeep',
+ 'john',
+ 'joker',
+ 'jordan',
+ 'jumbo',
+ 'june',
+ 'jungle',
+ 'junior',
+ 'jupiter',
+ 'karate',
+ 'karma',
+ 'kayak',
+ 'kermit',
+ 'kilo',
+ 'king',
+ 'koala',
+ 'korea',
+ 'labor',
+ 'lady',
+ 'lagoon',
+ 'laptop',
+ 'laser',
+ 'latin',
+ 'lava',
+ 'lecture',
+ 'left',
+ 'legal',
+ 'lemon',
+ 'level',
+ 'lexicon',
+ 'liberal',
+ 'libra',
+ 'limbo',
+ 'limit',
+ 'linda',
+ 'linear',
+ 'lion',
+ 'liquid',
+ 'liter',
+ 'little',
+ 'llama',
+ 'lobby',
+ 'lobster',
+ 'local',
+ 'logic',
+ 'logo',
+ 'lola',
+ 'london',
+ 'lotus',
+ 'lucas',
+ 'lunar',
+ 'machine',
+ 'macro',
+ 'madam',
+ 'madonna',
+ 'madrid',
+ 'maestro',
+ 'magic',
+ 'magnet',
+ 'magnum',
+ 'major',
+ 'mama',
+ 'mambo',
+ 'manager',
+ 'mango',
+ 'manila',
+ 'marco',
+ 'marina',
+ 'market',
+ 'mars',
+ 'martin',
+ 'marvin',
+ 'master',
+ 'matrix',
+ 'maximum',
+ 'media',
+ 'medical',
+ 'mega',
+ 'melody',
+ 'melon',
+ 'memo',
+ 'mental',
+ 'mentor',
+ 'menu',
+ 'mercury',
+ 'message',
+ 'metal',
+ 'meteor',
+ 'meter',
+ 'method',
+ 'metro',
+ 'mexico',
+ 'miami',
+ 'micro',
+ 'million',
+ 'mineral',
+ 'minimum',
+ 'minus',
+ 'minute',
+ 'miracle',
+ 'mirage',
+ 'miranda',
+ 'mister',
+ 'mixer',
+ 'mobile',
+ 'model',
+ 'modem',
+ 'modern',
+ 'modular',
+ 'moment',
+ 'monaco',
+ 'monica',
+ 'monitor',
+ 'mono',
+ 'monster',
+ 'montana',
+ 'morgan',
+ 'motel',
+ 'motif',
+ 'motor',
+ 'mozart',
+ 'multi',
+ 'museum',
+ 'music',
+ 'mustang',
+ 'natural',
+ 'neon',
+ 'nepal',
+ 'neptune',
+ 'nerve',
+ 'neutral',
+ 'nevada',
+ 'news',
+ 'ninja',
+ 'nirvana',
+ 'normal',
+ 'nova',
+ 'novel',
+ 'nuclear',
+ 'numeric',
+ 'nylon',
+ 'oasis',
+ 'object',
+ 'observe',
+ 'ocean',
+ 'octopus',
+ 'olivia',
+ 'olympic',
+ 'omega',
+ 'opera',
+ 'optic',
+ 'optimal',
+ 'orange',
+ 'orbit',
+ 'organic',
+ 'orient',
+ 'origin',
+ 'orlando',
+ 'oscar',
+ 'oxford',
+ 'oxygen',
+ 'ozone',
+ 'pablo',
+ 'pacific',
+ 'pagoda',
+ 'palace',
+ 'pamela',
+ 'panama',
+ 'panda',
+ 'panel',
+ 'panic',
+ 'paradox',
+ 'pardon',
+ 'paris',
+ 'parker',
+ 'parking',
+ 'parody',
+ 'partner',
+ 'passage',
+ 'passive',
+ 'pasta',
+ 'pastel',
+ 'patent',
+ 'patriot',
+ 'patrol',
+ 'patron',
+ 'pegasus',
+ 'pelican',
+ 'penguin',
+ 'pepper',
+ 'percent',
+ 'perfect',
+ 'perfume',
+ 'period',
+ 'permit',
+ 'person',
+ 'peru',
+ 'phone',
+ 'photo',
+ 'piano',
+ 'picasso',
+ 'picnic',
+ 'picture',
+ 'pigment',
+ 'pilgrim',
+ 'pilot',
+ 'pirate',
+ 'pixel',
+ 'pizza',
+ 'planet',
+ 'plasma',
+ 'plaster',
+ 'plastic',
+ 'plaza',
+ 'pocket',
+ 'poem',
+ 'poetic',
+ 'poker',
+ 'polaris',
+ 'police',
+ 'politic',
+ 'polo',
+ 'polygon',
+ 'pony',
+ 'popcorn',
+ 'popular',
+ 'postage',
+ 'postal',
+ 'precise',
+ 'prefix',
+ 'premium',
+ 'present',
+ 'price',
+ 'prince',
+ 'printer',
+ 'prism',
+ 'private',
+ 'product',
+ 'profile',
+ 'program',
+ 'project',
+ 'protect',
+ 'proton',
+ 'public',
+ 'pulse',
+ 'puma',
+ 'pyramid',
+ 'queen',
+ 'radar',
+ 'radio',
+ 'random',
+ 'rapid',
+ 'rebel',
+ 'record',
+ 'recycle',
+ 'reflex',
+ 'reform',
+ 'regard',
+ 'regular',
+ 'relax',
+ 'report',
+ 'reptile',
+ 'reverse',
+ 'ricardo',
+ 'ringo',
+ 'ritual',
+ 'robert',
+ 'robot',
+ 'rocket',
+ 'rodeo',
+ 'romeo',
+ 'royal',
+ 'russian',
+ 'safari',
+ 'salad',
+ 'salami',
+ 'salmon',
+ 'salon',
+ 'salute',
+ 'samba',
+ 'sandra',
+ 'santana',
+ 'sardine',
+ 'school',
+ 'screen',
+ 'script',
+ 'second',
+ 'secret',
+ 'section',
+ 'segment',
+ 'select',
+ 'seminar',
+ 'senator',
+ 'senior',
+ 'sensor',
+ 'serial',
+ 'service',
+ 'sheriff',
+ 'shock',
+ 'sierra',
+ 'signal',
+ 'silicon',
+ 'silver',
+ 'similar',
+ 'simon',
+ 'single',
+ 'siren',
+ 'slogan',
+ 'social',
+ 'soda',
+ 'solar',
+ 'solid',
+ 'solo',
+ 'sonic',
+ 'soviet',
+ 'special',
+ 'speed',
+ 'spiral',
+ 'spirit',
+ 'sport',
+ 'static',
+ 'station',
+ 'status',
+ 'stereo',
+ 'stone',
+ 'stop',
+ 'street',
+ 'strong',
+ 'student',
+ 'studio',
+ 'style',
+ 'subject',
+ 'sultan',
+ 'super',
+ 'susan',
+ 'sushi',
+ 'suzuki',
+ 'switch',
+ 'symbol',
+ 'system',
+ 'tactic',
+ 'tahiti',
+ 'talent',
+ 'tango',
+ 'tarzan',
+ 'taxi',
+ 'telex',
+ 'tempo',
+ 'tennis',
+ 'texas',
+ 'textile',
+ 'theory',
+ 'thermos',
+ 'tiger',
+ 'titanic',
+ 'tokyo',
+ 'tomato',
+ 'topic',
+ 'tornado',
+ 'toronto',
+ 'torpedo',
+ 'total',
+ 'totem',
+ 'tourist',
+ 'tractor',
+ 'traffic',
+ 'transit',
+ 'trapeze',
+ 'travel',
+ 'tribal',
+ 'trick',
+ 'trident',
+ 'trilogy',
+ 'tripod',
+ 'tropic',
+ 'trumpet',
+ 'tulip',
+ 'tuna',
+ 'turbo',
+ 'twist',
+ 'ultra',
+ 'uniform',
+ 'union',
+ 'uranium',
+ 'vacuum',
+ 'valid',
+ 'vampire',
+ 'vanilla',
+ 'vatican',
+ 'velvet',
+ 'ventura',
+ 'venus',
+ 'vertigo',
+ 'veteran',
+ 'victor',
+ 'video',
+ 'vienna',
+ 'viking',
+ 'village',
+ 'vincent',
+ 'violet',
+ 'violin',
+ 'virtual',
+ 'virus',
+ 'visa',
+ 'vision',
+ 'visitor',
+ 'visual',
+ 'vitamin',
+ 'viva',
+ 'vocal',
+ 'vodka',
+ 'volcano',
+ 'voltage',
+ 'volume',
+ 'voyage',
+ 'water',
+ 'weekend',
+ 'welcome',
+ 'western',
+ 'window',
+ 'winter',
+ 'wizard',
+ 'wolf',
+ 'world',
+ 'xray',
+ 'yankee',
+ 'yoga',
+ 'yogurt',
+ 'yoyo',
+ 'zebra',
+ 'zero',
+ 'zigzag',
+ 'zipper',
+ 'zodiac',
+ 'zoom',
+ 'abraham',
+ 'action',
+ 'address',
+ 'alabama',
+ 'alfred',
+ 'almond',
+ 'ammonia',
+ 'analyze',
+ 'annual',
+ 'answer',
+ 'apple',
+ 'arena',
+ 'armada',
+ 'arsenal',
+ 'atlanta',
+ 'atomic',
+ 'avenue',
+ 'average',
+ 'bagel',
+ 'baker',
+ 'ballet',
+ 'bambino',
+ 'bamboo',
+ 'barbara',
+ 'basket',
+ 'bazaar',
+ 'benefit',
+ 'bicycle',
+ 'bishop',
+ 'blitz',
+ 'bonjour',
+ 'bottle',
+ 'bridge',
+ 'british',
+ 'brother',
+ 'brush',
+ 'budget',
+ 'cabaret',
+ 'cadet',
+ 'candle',
+ 'capitan',
+ 'capsule',
+ 'career',
+ 'cartoon',
+ 'channel',
+ 'chapter',
+ 'cheese',
+ 'circle',
+ 'cobalt',
+ 'cockpit',
+ 'college',
+ 'compass',
+ 'comrade',
+ 'condor',
+ 'crimson',
+ 'cyclone',
+ 'darwin',
+ 'declare',
+ 'degree',
+ 'delete',
+ 'delphi',
+ 'denver',
+ 'desert',
+ 'divide',
+ 'dolby',
+ 'domain',
+ 'domingo',
+ 'double',
+ 'drink',
+ 'driver',
+ 'eagle',
+ 'earth',
+ 'echo',
+ 'eclipse',
+ 'editor',
+ 'educate',
+ 'edward',
+ 'effect',
+ 'electra',
+ 'emerald',
+ 'emotion',
+ 'empire',
+ 'empty',
+ 'escape',
+ 'eternal',
+ 'evening',
+ 'exhibit',
+ 'expand',
+ 'explore',
+ 'extreme',
+ 'ferrari',
+ 'first',
+ 'flag',
+ 'folio',
+ 'forget',
+ 'forward',
+ 'freedom',
+ 'fresh',
+ 'friday',
+ 'fuji',
+ 'galileo',
+ 'garcia',
+ 'genesis',
+ 'gold',
+ 'gravity',
+ 'habitat',
+ 'hamlet',
+ 'harlem',
+ 'helium',
+ 'holiday',
+ 'house',
+ 'hunter',
+ 'ibiza',
+ 'iceberg',
+ 'imagine',
+ 'infant',
+ 'isotope',
+ 'jackson',
+ 'jamaica',
+ 'jasmine',
+ 'java',
+ 'jessica',
+ 'judo',
+ 'kitchen',
+ 'lazarus',
+ 'letter',
+ 'license',
+ 'lithium',
+ 'loyal',
+ 'lucky',
+ 'magenta',
+ 'mailbox',
+ 'manual',
+ 'marble',
+ 'mary',
+ 'maxwell',
+ 'mayor',
+ 'milk',
+ 'monarch',
+ 'monday',
+ 'money',
+ 'morning',
+ 'mother',
+ 'mystery',
+ 'native',
+ 'nectar',
+ 'nelson',
+ 'network',
+ 'next',
+ 'nikita',
+ 'nobel',
+ 'nobody',
+ 'nominal',
+ 'norway',
+ 'nothing',
+ 'number',
+ 'october',
+ 'office',
+ 'oliver',
+ 'opinion',
+ 'option',
+ 'order',
+ 'outside',
+ 'package',
+ 'pancake',
+ 'pandora',
+ 'panther',
+ 'papa',
+ 'patient',
+ 'pattern',
+ 'pedro',
+ 'pencil',
+ 'people',
+ 'phantom',
+ 'philips',
+ 'pioneer',
+ 'pluto',
+ 'podium',
+ 'portal',
+ 'potato',
+ 'prize',
+ 'process',
+ 'protein',
+ 'proxy',
+ 'pump',
+ 'pupil',
+ 'python',
+ 'quality',
+ 'quarter',
+ 'quiet',
+ 'rabbit',
+ 'radical',
+ 'radius',
+ 'rainbow',
+ 'ralph',
+ 'ramirez',
+ 'ravioli',
+ 'raymond',
+ 'respect',
+ 'respond',
+ 'result',
+ 'resume',
+ 'retro',
+ 'richard',
+ 'right',
+ 'risk',
+ 'river',
+ 'roger',
+ 'roman',
+ 'rondo',
+ 'sabrina',
+ 'salary',
+ 'salsa',
+ 'sample',
+ 'samuel',
+ 'saturn',
+ 'savage',
+ 'scarlet',
+ 'scoop',
+ 'scorpio',
+ 'scratch',
+ 'scroll',
+ 'sector',
+ 'serpent',
+ 'shadow',
+ 'shampoo',
+ 'sharon',
+ 'sharp',
+ 'short',
+ 'shrink',
+ 'silence',
+ 'silk',
+ 'simple',
+ 'slang',
+ 'smart',
+ 'smoke',
+ 'snake',
+ 'society',
+ 'sonar',
+ 'sonata',
+ 'soprano',
+ 'source',
+ 'sparta',
+ 'sphere',
+ 'spider',
+ 'sponsor',
+ 'spring',
+ 'acid',
+ 'adios',
+ 'agatha',
+ 'alamo',
+ 'alert',
+ 'almanac',
+ 'aloha',
+ 'andrea',
+ 'anita',
+ 'arcade',
+ 'aurora',
+ 'avalon',
+ 'baby',
+ 'baggage',
+ 'balloon',
+ 'bank',
+ 'basil',
+ 'begin',
+ 'biscuit',
+ 'blue',
+ 'bombay',
+ 'brain',
+ 'brenda',
+ 'brigade',
+ 'cable',
+ 'carmen',
+ 'cello',
+ 'celtic',
+ 'chariot',
+ 'chrome',
+ 'citrus',
+ 'civil',
+ 'cloud',
+ 'common',
+ 'compare',
+ 'cool',
+ 'copper',
+ 'coral',
+ 'crater',
+ 'cubic',
+ 'cupid',
+ 'cycle',
+ 'depend',
+ 'door',
+ 'dream',
+ 'dynasty',
+ 'edison',
+ 'edition',
+ 'enigma',
+ 'equal',
+ 'eric',
+ 'event',
+ 'evita',
+ 'exodus',
+ 'extend',
+ 'famous',
+ 'farmer',
+ 'food',
+ 'fossil',
+ 'frog',
+ 'fruit',
+ 'geneva',
+ 'gentle',
+ 'george',
+ 'giant',
+ 'gilbert',
+ 'gossip',
+ 'gram',
+ 'greek',
+ 'grille',
+ 'hammer',
+ 'harvest',
+ 'hazard',
+ 'heaven',
+ 'herbert',
+ 'heroic',
+ 'hexagon',
+ 'husband',
+ 'immune',
+ 'inca',
+ 'inch',
+ 'initial',
+ 'isabel',
+ 'ivory',
+ 'jason',
+ 'jerome',
+ 'joel',
+ 'joshua',
+ 'journal',
+ 'judge',
+ 'juliet',
+ 'jump',
+ 'justice',
+ 'kimono',
+ 'kinetic',
+ 'leonid',
+ 'lima',
+ 'maze',
+ 'medusa',
+ 'member',
+ 'memphis',
+ 'michael',
+ 'miguel',
+ 'milan',
+ 'mile',
+ 'miller',
+ 'mimic',
+ 'mimosa',
+ 'mission',
+ 'monkey',
+ 'moral',
+ 'moses',
+ 'mouse',
+ 'nancy',
+ 'natasha',
+ 'nebula',
+ 'nickel',
+ 'nina',
+ 'noise',
+ 'orchid',
+ 'oregano',
+ 'origami',
+ 'orinoco',
+ 'orion',
+ 'othello',
+ 'paper',
+ 'paprika',
+ 'prelude',
+ 'prepare',
+ 'pretend',
+ 'profit',
+ 'promise',
+ 'provide',
+ 'puzzle',
+ 'remote',
+ 'repair',
+ 'reply',
+ 'rival',
+ 'riviera',
+ 'robin',
+ 'rose',
+ 'rover',
+ 'rudolf',
+ 'saga',
+ 'sahara',
+ 'scholar',
+ 'shelter',
+ 'ship',
+ 'shoe',
+ 'sigma',
+ 'sister',
+ 'sleep',
+ 'smile',
+ 'spain',
+ 'spark',
+ 'split',
+ 'spray',
+ 'square',
+ 'stadium',
+ 'star',
+ 'storm',
+ 'story',
+ 'strange',
+ 'stretch',
+ 'stuart',
+ 'subway',
+ 'sugar',
+ 'sulfur',
+ 'summer',
+ 'survive',
+ 'sweet',
+ 'swim',
+ 'table',
+ 'taboo',
+ 'target',
+ 'teacher',
+ 'telecom',
+ 'temple',
+ 'tibet',
+ 'ticket',
+ 'tina',
+ 'today',
+ 'toga',
+ 'tommy',
+ 'tower',
+ 'trivial',
+ 'tunnel',
+ 'turtle',
+ 'twin',
+ 'uncle',
+ 'unicorn',
+ 'unique',
+ 'update',
+ 'valery',
+ 'vega',
+ 'version',
+ 'voodoo',
+ 'warning',
+ 'william',
+ 'wonder',
+ 'year',
+ 'yellow',
+ 'young',
+ 'absent',
+ 'absorb',
+ 'accent',
+ 'alfonso',
+ 'alias',
+ 'ambient',
+ 'andy',
+ 'anvil',
+ 'appear',
+ 'apropos',
+ 'archer',
+ 'ariel',
+ 'armor',
+ 'arrow',
+ 'austin',
+ 'avatar',
+ 'axis',
+ 'baboon',
+ 'bahama',
+ 'bali',
+ 'balsa',
+ 'bazooka',
+ 'beach',
+ 'beast',
+ 'beatles',
+ 'beauty',
+ 'before',
+ 'benny',
+ 'betty',
+ 'between',
+ 'beyond',
+ 'billy',
+ 'bison',
+ 'blast',
+ 'bless',
+ 'bogart',
+ 'bonanza',
+ 'book',
+ 'border',
+ 'brave',
+ 'bread',
+ 'break',
+ 'broken',
+ 'bucket',
+ 'buenos',
+ 'buffalo',
+ 'bundle',
+ 'button',
+ 'buzzer',
+ 'byte',
+ 'caesar',
+ 'camilla',
+ 'canary',
+ 'candid',
+ 'carrot',
+ 'cave',
+ 'chant',
+ 'child',
+ 'choice',
+ 'chris',
+ 'cipher',
+ 'clarion',
+ 'clark',
+ 'clever',
+ 'cliff',
+ 'clone',
+ 'conan',
+ 'conduct',
+ 'congo',
+ 'content',
+ 'costume',
+ 'cotton',
+ 'cover',
+ 'crack',
+ 'current',
+ 'danube',
+ 'data',
+ 'decide',
+ 'desire',
+ 'detail',
+ 'dexter',
+ 'dinner',
+ 'dispute',
+ 'donor',
+ 'druid',
+ 'drum',
+ 'easy',
+ 'eddie',
+ 'enjoy',
+ 'enrico',
+ 'epoxy',
+ 'erosion',
+ 'except',
+ 'exile',
+ 'explain',
+ 'fame',
+ 'fast',
+ 'father',
+ 'felix',
+ 'field',
+ 'fiona',
+ 'fire',
+ 'fish',
+ 'flame',
+ 'flex',
+ 'flipper',
+ 'float',
+ 'flood',
+ 'floor',
+ 'forbid',
+ 'forever',
+ 'fractal',
+ 'frame',
+ 'freddie',
+ 'front',
+ 'fuel',
+ 'gallop',
+ 'game',
+ 'garbo',
+ 'gate',
+ 'gibson',
+ 'ginger',
+ 'giraffe',
+ 'gizmo',
+ 'glass',
+ 'goblin',
+ 'gopher',
+ 'grace',
+ 'gray',
+ 'gregory',
+ 'grid',
+ 'griffin',
+ 'ground',
+ 'guest',
+ 'gustav',
+ 'gyro',
+ 'hair',
+ 'halt',
+ 'harris',
+ 'heart',
+ 'heavy',
+ 'herman',
+ 'hippie',
+ 'hobby',
+ 'honey',
+ 'hope',
+ 'horse',
+ 'hostel',
+ 'hydro',
+ 'imitate',
+ 'info',
+ 'ingrid',
+ 'inside',
+ 'invent',
+ 'invest',
+ 'invite',
+ 'iron',
+ 'ivan',
+ 'james',
+ 'jester',
+ 'jimmy',
+ 'join',
+ 'joseph',
+ 'juice',
+ 'julius',
+ 'july',
+ 'justin',
+ 'kansas',
+ 'karl',
+ 'kevin',
+ 'kiwi',
+ 'ladder',
+ 'lake',
+ 'laura',
+ 'learn',
+ 'legacy',
+ 'legend',
+ 'lesson',
+ 'life',
+ 'light',
+ 'list',
+ 'locate',
+ 'lopez',
+ 'lorenzo',
+ 'love',
+ 'lunch',
+ 'malta',
+ 'mammal',
+ 'margo',
+ 'marion',
+ 'mask',
+ 'match',
+ 'mayday',
+ 'meaning',
+ 'mercy',
+ 'middle',
+ 'mike',
+ 'mirror',
+ 'modest',
+ 'morph',
+ 'morris',
+ 'nadia',
+ 'nato',
+ 'navy',
+ 'needle',
+ 'neuron',
+ 'never',
+ 'newton',
+ 'nice',
+ 'night',
+ 'nissan',
+ 'nitro',
+ 'nixon',
+ 'north',
+ 'oberon',
+ 'octavia',
+ 'ohio',
+ 'olga',
+ 'open',
+ 'opus',
+ 'orca',
+ 'oval',
+ 'owner',
+ 'page',
+ 'paint',
+ 'palma',
+ 'parade',
+ 'parent',
+ 'parole',
+ 'paul',
+ 'peace',
+ 'pearl',
+ 'perform',
+ 'phoenix',
+ 'phrase',
+ 'pierre',
+ 'pinball',
+ 'place',
+ 'plate',
+ 'plato',
+ 'plume',
+ 'pogo',
+ 'point',
+ 'polite',
+ 'polka',
+ 'poncho',
+ 'powder',
+ 'prague',
+ 'press',
+ 'presto',
+ 'pretty',
+ 'prime',
+ 'promo',
+ 'quasi',
+ 'quest',
+ 'quick',
+ 'quiz',
+ 'quota',
+ 'race',
+ 'rachel',
+ 'raja',
+ 'ranger',
+ 'region',
+ 'remark',
+ 'rent',
+ 'reward',
+ 'rhino',
+ 'ribbon',
+ 'rider',
+ 'road',
+ 'rodent',
+ 'round',
+ 'rubber',
+ 'ruby',
+ 'rufus',
+ 'sabine',
+ 'saddle',
+ 'sailor',
+ 'saint',
+ 'salt',
+ 'satire',
+ 'scale',
+ 'scuba',
+ 'season',
+ 'secure',
+ 'shake',
+ 'shallow',
+ 'shannon',
+ 'shave',
+ 'shelf',
+ 'sherman',
+ 'shine',
+ 'shirt',
+ 'side',
+ 'sinatra',
+ 'sincere',
+ 'size',
+ 'slalom',
+ 'slow',
+ 'small',
+ 'snow',
+ 'sofia',
+ 'song',
+ 'sound',
+ 'south',
+ 'speech',
+ 'spell',
+ 'spend',
+ 'spoon',
+ 'stage',
+ 'stamp',
+ 'stand',
+ 'state',
+ 'stella',
+ 'stick',
+ 'sting',
+ 'stock',
+ 'store',
+ 'sunday',
+ 'sunset',
+ 'support',
+ 'sweden',
+ 'swing',
+ 'tape',
+ 'think',
+ 'thomas',
+ 'tictac',
+ 'time',
+ 'toast',
+ 'tobacco',
+ 'tonight',
+ 'torch',
+ 'torso',
+ 'touch',
+ 'toyota',
+ 'trade',
+ 'tribune',
+ 'trinity',
+ 'triton',
+ 'truck',
+ 'trust',
+ 'type',
+ 'under',
+ 'unit',
+ 'urban',
+ 'urgent',
+ 'user',
+ 'value',
+ 'vendor',
+ 'venice',
+ 'verona',
+ 'vibrate',
+ 'virgo',
+ 'visible',
+ 'vista',
+ 'vital',
+ 'voice',
+ 'vortex',
+ 'waiter',
+ 'watch',
+ 'wave',
+ 'weather',
+ 'wedding',
+ 'wheel',
+ 'whiskey',
+ 'wisdom',
+ 'deal',
+ 'null',
+ 'nurse',
+ 'quebec',
+ 'reserve',
+ 'reunion',
+ 'roof',
+ 'singer',
+ 'verbal',
+ 'amen',
+ 'ego',
+ 'fax',
+ 'jet',
+ 'job',
+ 'rio',
+ 'ski',
+ 'yes',
];
diff --git a/public/styles/pin.sass b/public/styles/pin.sass
index 614dae0..56be1b4 100644
--- a/public/styles/pin.sass
+++ b/public/styles/pin.sass
@@ -6,43 +6,43 @@
/
.columns
- min-height: 100%
+ min-height: 100%
.enter-pin
- position: relative
- width: 80vw
- margin: 0 auto
- top: 50%
- transform: translateY(-50%)
+ position: relative
+ width: 80vw
+ margin: 0 auto
+ top: 50%
+ transform: translateY(-50%)
.pin-wrapper
- padding: 0
- margin: 0
- align-items: center
- justify-content: center
+ padding: 0
+ margin: 0
+ align-items: center
+ justify-content: center
.pin-wrapper > .column
- align-items: center
- justify-content: center
+ align-items: center
+ justify-content: center
.pin-wrapper > .column input
- width: 15vw
- height: 15vw
- font-size: 15vw
- background-color: #2d3338
- color: transparent
- text-shadow: 0 0 0 #000
- border: none
- border-radius: 10px
- text-align: center
+ width: 15vw
+ height: 15vw
+ font-size: 15vw
+ background-color: #2d3338
+ color: transparent
+ text-shadow: 0 0 0 #000
+ border: none
+ border-radius: 10px
+ text-align: center
.pin-wrapper > .column input::-webkit-outer-spin-button, .pin-wrapper > .column input::-webkit-inner-spin-button
- -webkit-appearance: none
- margin: 0
+ -webkit-appearance: none
+ margin: 0
.pin-wrapper > .column input:focus
- background-color: #394046
- border-color: inherit
- -webkit-box-shadow: none
- box-shadow: none
- outline: none
+ background-color: #394046
+ border-color: inherit
+ -webkit-box-shadow: none
+ box-shadow: none
+ outline: none
diff --git a/public/styles/style.sass b/public/styles/style.sass
index a7feeac..7f489e2 100644
--- a/public/styles/style.sass
+++ b/public/styles/style.sass
@@ -6,53 +6,53 @@
/
html, body
- height: 100% !important
- width: 100% !important
- overflow: hidden !important
- background-color: #23272a !important
+ height: 100% !important
+ width: 100% !important
+ overflow: hidden !important
+ background-color: #23272a !important
.chat
- display: none
+ display: none
.main
- align-items: flex-start
+ align-items: flex-start
.swal-text
- text-align: center
+ text-align: center
.swal-modal, .swal-title, .swal-text, .swal-icon--success__hide-corners, .swal-icon--success:after, .swal-icon--success:before
- color: #fff
- background: #293034
+ color: #fff
+ background: #293034
.swal-icon--success__line
- background-color: #00940a !important
+ background-color: #00940a !important
.swal-icon--success__ring
- border: 4px solid hsla(131, 97%, 44%, 0.2)
+ border: 4px solid hsla(131, 97%, 44%, 0.2)
.action-button
- height: 50px
- width: 50px
- margin: 20px
- border-radius: 100%
- transition: background-color .3s linear
+ height: 50px
+ width: 50px
+ margin: 20px
+ border-radius: 100%
+ transition: background-color .3s linear
.message-input
- width: 100%
- padding-right: 50px
- color: #fff
- background-color: #2d3338
- border-color: #2d3338
+ width: 100%
+ padding-right: 50px
+ color: #fff
+ background-color: #2d3338
+ border-color: #2d3338
.message-input::placeholder
- color: #909090
+ color: #909090
.message-button
- width: 50px
- margin-left: -50px
+ width: 50px
+ margin-left: -50px
.message-field
- position: absolute
- width: 100%
- bottom: 0
- left: 0
+ position: absolute
+ width: 100%
+ bottom: 0
+ left: 0