|
|
@ -4,7 +4,10 @@ const express = require('express'),
|
|
|
|
mdEmoji = require('markdown-it-emoji'),
|
|
|
|
mdEmoji = require('markdown-it-emoji'),
|
|
|
|
mdMark = require('markdown-it-mark'),
|
|
|
|
mdMark = require('markdown-it-mark'),
|
|
|
|
mdSmartarrows = require('markdown-it-smartarrows'),
|
|
|
|
mdSmartarrows = require('markdown-it-smartarrows'),
|
|
|
|
md = require('markdown-it')()
|
|
|
|
md = require('markdown-it')({
|
|
|
|
|
|
|
|
linkify: true,
|
|
|
|
|
|
|
|
typographer: true
|
|
|
|
|
|
|
|
})
|
|
|
|
.use(mdEmoji)
|
|
|
|
.use(mdEmoji)
|
|
|
|
.use(mdMark)
|
|
|
|
.use(mdMark)
|
|
|
|
.use(mdSmartarrows),
|
|
|
|
.use(mdSmartarrows),
|
|
|
@ -12,6 +15,7 @@ const express = require('express'),
|
|
|
|
globals = require('../lib/globals');
|
|
|
|
globals = require('../lib/globals');
|
|
|
|
|
|
|
|
|
|
|
|
let pgPool = globals.pgPool;
|
|
|
|
let pgPool = globals.pgPool;
|
|
|
|
|
|
|
|
let sockets = {};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Class to manage the bingo data in the database.
|
|
|
|
* Class to manage the bingo data in the database.
|
|
|
@ -138,7 +142,16 @@ class BingoDataManager {
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
async updateLobbyExpiration(lobbyId) {
|
|
|
|
async updateLobbyExpiration(lobbyId) {
|
|
|
|
return await this._queryDatabase(this.queries.updateLobbyExpire.sql, [lobbyId]);
|
|
|
|
return await this._queryFirstResult(this.queries.updateLobbyExpire.sql, [lobbyId]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Returns all lobby ids
|
|
|
|
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async getLobbyIds() {
|
|
|
|
|
|
|
|
let results = await this._queryAllResults(this.queries.getLobbyIds.sql, []);
|
|
|
|
|
|
|
|
return results.map(x => x.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
@ -430,6 +443,34 @@ class BingoDataManager {
|
|
|
|
return await this._queryFirstResult(this.queries.addUserMessage.sql, [playerId, lobbyId, messageContent]);
|
|
|
|
return await this._queryFirstResult(this.queries.addUserMessage.sql, [playerId, lobbyId, messageContent]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Edits a message
|
|
|
|
|
|
|
|
* @param messageId {Number} - the id of the message
|
|
|
|
|
|
|
|
* @param messageContent {String} - the new content of the message
|
|
|
|
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async editMessage(messageId, messageContent) {
|
|
|
|
|
|
|
|
return await this._queryFirstResult(this.queries.editMessage.sql, [messageId, messageContent]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Deletes a message
|
|
|
|
|
|
|
|
* @param messageId {Number} - the id of the message
|
|
|
|
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async deleteMessage(messageId) {
|
|
|
|
|
|
|
|
return await this._queryFirstResult(this.queries.deleteMessage.sql, [messageId]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Returns the data of a message
|
|
|
|
|
|
|
|
* @param messageId {Number} - the id of the message
|
|
|
|
|
|
|
|
* @returns {Promise<*>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async getMessageData(messageId) {
|
|
|
|
|
|
|
|
return await this._queryFirstResult(this.queries.getMessageData.sql, [messageId]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Adds a message of type "INFO" to the lobby
|
|
|
|
* Adds a message of type "INFO" to the lobby
|
|
|
|
* @param lobbyId {Number} - the id of the lobby
|
|
|
|
* @param lobbyId {Number} - the id of the lobby
|
|
|
@ -650,12 +691,12 @@ class GridWrapper {
|
|
|
|
let gridField = new GridFieldWrapper(result);
|
|
|
|
let gridField = new GridFieldWrapper(result);
|
|
|
|
let username = await (await this.player()).username();
|
|
|
|
let username = await (await this.player()).username();
|
|
|
|
let word = await gridField.word.content();
|
|
|
|
let word = await gridField.word.content();
|
|
|
|
|
|
|
|
let lobbyWrapper = await this.lobby();
|
|
|
|
|
|
|
|
|
|
|
|
if (gridField.submitted)
|
|
|
|
if (gridField.submitted)
|
|
|
|
await bdm.addInfoMessage(this.lobbyId,
|
|
|
|
await lobbyWrapper.addInfoMessage(`${username} toggled "${word}"`);
|
|
|
|
`${username} toggled "${word}"`);
|
|
|
|
|
|
|
|
else
|
|
|
|
else
|
|
|
|
await bdm.addInfoMessage(this.lobbyId,
|
|
|
|
await lobbyWrapper.addInfoMessage(`${username} untoggled "${word}"`);
|
|
|
|
`${username} untoggled "${word}"`);
|
|
|
|
|
|
|
|
return gridField;
|
|
|
|
return gridField;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -668,7 +709,7 @@ class MessageWrapper {
|
|
|
|
constructor(row) {
|
|
|
|
constructor(row) {
|
|
|
|
this.id = row.id;
|
|
|
|
this.id = row.id;
|
|
|
|
this.content = row.content;
|
|
|
|
this.content = row.content;
|
|
|
|
this.htmlContent = md.renderInline(this.content);
|
|
|
|
this.htmlContent = md.renderInline(preMarkdownParse(this.content));
|
|
|
|
this.author = new PlayerWrapper(row.player_id);
|
|
|
|
this.author = new PlayerWrapper(row.player_id);
|
|
|
|
this.lobby = new LobbyWrapper(row.lobby_id);
|
|
|
|
this.lobby = new LobbyWrapper(row.lobby_id);
|
|
|
|
this.type = row.type;
|
|
|
|
this.type = row.type;
|
|
|
@ -881,6 +922,7 @@ class RoundWrapper {
|
|
|
|
let updateResult = await bdm.setRoundWinner(this.id, winnerId);
|
|
|
|
let updateResult = await bdm.setRoundWinner(this.id, winnerId);
|
|
|
|
if (updateResult)
|
|
|
|
if (updateResult)
|
|
|
|
await this.setFinished();
|
|
|
|
await this.setFinished();
|
|
|
|
|
|
|
|
(await this.lobby()).socket.emit('statusChange', 'FINISHED', await resolvePlayer(new PlayerWrapper(winnerId)));
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -894,6 +936,7 @@ class LobbyWrapper {
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
constructor(id, row) {
|
|
|
|
constructor(id, row) {
|
|
|
|
this.id = id;
|
|
|
|
this.id = id;
|
|
|
|
|
|
|
|
this.socket = sockets[id];
|
|
|
|
this._infoLoaded = false;
|
|
|
|
this._infoLoaded = false;
|
|
|
|
if (row)
|
|
|
|
if (row)
|
|
|
|
this._assignProperties(row);
|
|
|
|
this._assignProperties(row);
|
|
|
@ -907,7 +950,7 @@ class LobbyWrapper {
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
async _loadLobbyInfo(force) {
|
|
|
|
async _loadLobbyInfo(force) {
|
|
|
|
if (!this._infoLoaded && !force) {
|
|
|
|
if (!this._infoLoaded && !force) {
|
|
|
|
let row = await bdm.getLobbyInfo(this.id);
|
|
|
|
let row = await bdm.updateLobbyExpiration(this.id);
|
|
|
|
this._assignProperties(row);
|
|
|
|
this._assignProperties(row);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -928,6 +971,14 @@ class LobbyWrapper {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Emits an event is a socket exists for the lobby
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
emit() {
|
|
|
|
|
|
|
|
if (this.socket)
|
|
|
|
|
|
|
|
this.socket.emit(...arguments);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Returns if the lobby exists (based on one loaded attribute)
|
|
|
|
* Returns if the lobby exists (based on one loaded attribute)
|
|
|
|
* @returns {Promise<boolean>}
|
|
|
|
* @returns {Promise<boolean>}
|
|
|
@ -1061,6 +1112,7 @@ class LobbyWrapper {
|
|
|
|
await this._createRound();
|
|
|
|
await this._createRound();
|
|
|
|
await this._createGrids();
|
|
|
|
await this._createGrids();
|
|
|
|
await this.setRoundStatus('ACTIVE');
|
|
|
|
await this.setRoundStatus('ACTIVE');
|
|
|
|
|
|
|
|
this.emit('statusChange', 'ACTIVE');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -1115,6 +1167,7 @@ class LobbyWrapper {
|
|
|
|
await this.addWord(word);
|
|
|
|
await this.addWord(word);
|
|
|
|
for (let word of removedWords)
|
|
|
|
for (let word of removedWords)
|
|
|
|
await this.removeWord(word.id);
|
|
|
|
await this.removeWord(word.id);
|
|
|
|
|
|
|
|
this.emit('wordsChange');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -1144,6 +1197,16 @@ class LobbyWrapper {
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Adds an info message and emits the message event.
|
|
|
|
|
|
|
|
* @param message {String} - the info messages content
|
|
|
|
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async addInfoMessage(message) {
|
|
|
|
|
|
|
|
let result = await bdm.addInfoMessage(this.id, message);
|
|
|
|
|
|
|
|
this.emit('message', await resolveMessage(new MessageWrapper(result)));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Adds a player to the lobby.
|
|
|
|
* Adds a player to the lobby.
|
|
|
|
* @param playerId
|
|
|
|
* @param playerId
|
|
|
@ -1151,8 +1214,10 @@ class LobbyWrapper {
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
async addPlayer(playerId) {
|
|
|
|
async addPlayer(playerId) {
|
|
|
|
await bdm.addPlayerToLobby(playerId, this.id);
|
|
|
|
await bdm.addPlayerToLobby(playerId, this.id);
|
|
|
|
let username = await new PlayerWrapper(playerId).username();
|
|
|
|
let playerWrapper = new PlayerWrapper(playerId);
|
|
|
|
await bdm.addInfoMessage(this.id, `${username} joined.`);
|
|
|
|
this.emit('playerJoin', await resolvePlayer(playerWrapper));
|
|
|
|
|
|
|
|
let username = await playerWrapper.username();
|
|
|
|
|
|
|
|
await this.addInfoMessage(`${username} joined.`);
|
|
|
|
await this._loadLobbyInfo(true);
|
|
|
|
await this._loadLobbyInfo(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -1164,7 +1229,8 @@ class LobbyWrapper {
|
|
|
|
async removePlayer(playerId) {
|
|
|
|
async removePlayer(playerId) {
|
|
|
|
await bdm.removePlayerFromLobby(playerId, this.id);
|
|
|
|
await bdm.removePlayerFromLobby(playerId, this.id);
|
|
|
|
let username = await new PlayerWrapper(playerId).username();
|
|
|
|
let username = await new PlayerWrapper(playerId).username();
|
|
|
|
await bdm.addInfoMessage(this.id, `${username} left.`);
|
|
|
|
this.emit('playerLeave', playerId);
|
|
|
|
|
|
|
|
await this.addInfoMessage(`${username} left.`);
|
|
|
|
await this._loadLobbyInfo(true);
|
|
|
|
await this._loadLobbyInfo(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -1185,7 +1251,8 @@ class LobbyWrapper {
|
|
|
|
async setRoundStatus(status) {
|
|
|
|
async setRoundStatus(status) {
|
|
|
|
let currentRound = await this.currentRound();
|
|
|
|
let currentRound = await this.currentRound();
|
|
|
|
await currentRound.updateStatus(status);
|
|
|
|
await currentRound.updateStatus(status);
|
|
|
|
await bdm.addInfoMessage(this.id, `Admin set round status to ${status}`);
|
|
|
|
await this.addInfoMessage(`Admin set round status to ${status}`);
|
|
|
|
|
|
|
|
this.emit('statusChange', status);
|
|
|
|
|
|
|
|
|
|
|
|
if (status === 'FINISHED')
|
|
|
|
if (status === 'FINISHED')
|
|
|
|
await bdm.clearGrids(this.id);
|
|
|
|
await bdm.clearGrids(this.id);
|
|
|
@ -1333,6 +1400,28 @@ function checkBingo(fg) {
|
|
|
|
return diagonalBingo || verticalCheck || horizontalCheck;
|
|
|
|
return diagonalBingo || verticalCheck || horizontalCheck;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Parses the message and replaces all links with markdown-links and images with markdown-images.
|
|
|
|
|
|
|
|
* @param message {String} - the raw message
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
function preMarkdownParse(message) {
|
|
|
|
|
|
|
|
let linkMatch = /(^|[^(])https?:\/\/((([\w-]+\.)+[\w-]+)(\S*))([^)]|$)/g;
|
|
|
|
|
|
|
|
let imageMatch = /.*\.(\w+)/g;
|
|
|
|
|
|
|
|
let imageExtensions = ['png', 'jpg', 'jpeg', 'gif', 'bmp', 'svg'];
|
|
|
|
|
|
|
|
let links = message.match(linkMatch);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (links)
|
|
|
|
|
|
|
|
for (let link of links) {
|
|
|
|
|
|
|
|
let linkGroups = linkMatch.exec(link);
|
|
|
|
|
|
|
|
let imgGroups = imageMatch.exec(link);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (imgGroups && imgGroups[1] && imageExtensions.includes(imgGroups[1]))
|
|
|
|
|
|
|
|
message = message.replace(link, `![${linkGroups[1]}](${link})`);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return message;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Gets player data for a lobby
|
|
|
|
* Gets player data for a lobby
|
|
|
|
* @param lobbyWrapper
|
|
|
|
* @param lobbyWrapper
|
|
|
@ -1395,6 +1484,53 @@ async function getGridData(lobbyId, playerId) {
|
|
|
|
return {fields: fieldGrid, bingo: await grid.bingo()};
|
|
|
|
return {fields: fieldGrid, bingo: await grid.bingo()};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Resolves a message wrapper object
|
|
|
|
|
|
|
|
* @param msgWrapper
|
|
|
|
|
|
|
|
* @returns {Promise<{author: {id: (*|MessageWrapper.author.id), username: String}, id: *, content: *, timestamp: Timestamp | * | number, htmlContent: *}>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async function resolveMessage(msgWrapper) {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
id: msgWrapper.id,
|
|
|
|
|
|
|
|
type: msgWrapper.type,
|
|
|
|
|
|
|
|
content: msgWrapper.content,
|
|
|
|
|
|
|
|
timestamp: msgWrapper.timestamp,
|
|
|
|
|
|
|
|
htmlContent: msgWrapper.htmlContent,
|
|
|
|
|
|
|
|
author: {
|
|
|
|
|
|
|
|
id: msgWrapper.author.id,
|
|
|
|
|
|
|
|
username: await msgWrapper.author.username()
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Resolves a player wrapper object
|
|
|
|
|
|
|
|
* @param playerWrapper
|
|
|
|
|
|
|
|
* @param lobbyId
|
|
|
|
|
|
|
|
* @returns {Promise<{wins: PlayerWrapper.wins, id: *, username: (String|*)}>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async function resolvePlayer(playerWrapper, lobbyId) {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
id: playerWrapper.id,
|
|
|
|
|
|
|
|
username: await playerWrapper.username(),
|
|
|
|
|
|
|
|
wins: await playerWrapper.wins({lobbyId: lobbyId})
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Resolves a fieldWrapper object
|
|
|
|
|
|
|
|
* @param fieldWrapper
|
|
|
|
|
|
|
|
* @returns {Promise<{submitted: (Object.submitted|*), column: *, bingo: boolean, row: (*)}>}
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
async function resolveGridField(fieldWrapper) {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
row: fieldWrapper.row,
|
|
|
|
|
|
|
|
column: fieldWrapper.column,
|
|
|
|
|
|
|
|
submitted: fieldWrapper.submitted,
|
|
|
|
|
|
|
|
bingo: await fieldWrapper.grid.bingo()
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Returns resolved message data.
|
|
|
|
* Returns resolved message data.
|
|
|
|
* @param lobbyId
|
|
|
|
* @param lobbyId
|
|
|
@ -1405,26 +1541,87 @@ async function getMessageData(lobbyId) {
|
|
|
|
let messages = await lobbyWrapper.messages({limit: 20});
|
|
|
|
let messages = await lobbyWrapper.messages({limit: 20});
|
|
|
|
let msgReturn = [];
|
|
|
|
let msgReturn = [];
|
|
|
|
for (let message of messages)
|
|
|
|
for (let message of messages)
|
|
|
|
msgReturn.push(Object.assign(message, {username: await message.author.username()}));
|
|
|
|
msgReturn.push(Object.assign(message, {
|
|
|
|
|
|
|
|
playerId: message.author.id,
|
|
|
|
|
|
|
|
username: await message.author.username()
|
|
|
|
|
|
|
|
}));
|
|
|
|
return msgReturn;
|
|
|
|
return msgReturn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// -- Router stuff
|
|
|
|
// -- Router stuff
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Creates a lobby socket if none exists.
|
|
|
|
|
|
|
|
* @param io
|
|
|
|
|
|
|
|
* @param lobbyId
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
function createSocketIfNotExist(io, lobbyId) {
|
|
|
|
|
|
|
|
if (!sockets[lobbyId]) {
|
|
|
|
|
|
|
|
let lobbySocket = io.of(`/bingo/${lobbyId}`);
|
|
|
|
|
|
|
|
sockets[lobbyId] = lobbySocket;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
lobbySocket.on('connection', (socket) => {
|
|
|
|
|
|
|
|
socket.on('message', async (context, message) => {
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
|
|
let result = await bdm.addUserMessage(lobbyId, context.playerId, message);
|
|
|
|
|
|
|
|
let messageWrapper = new MessageWrapper(result);
|
|
|
|
|
|
|
|
lobbySocket.emit('message', await resolveMessage(messageWrapper));
|
|
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
|
|
console.error(err);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
socket.on('messageEdit', async (context, message, messageId) => {
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
|
|
let row = await bdm.getMessageData(messageId);
|
|
|
|
|
|
|
|
if (row.player_id === Number(context.playerId)) {
|
|
|
|
|
|
|
|
let result = await bdm.editMessage(messageId, message);
|
|
|
|
|
|
|
|
let messageWrapper = new MessageWrapper(result);
|
|
|
|
|
|
|
|
lobbySocket.emit('messageEdit', await resolveMessage(messageWrapper));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
socket.emit('userError', "You are only allowed to edit your messages.");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
|
|
console.error(err);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
socket.on('messageDelete', async (context, messageId) => {
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
|
|
let row = await bdm.getMessageData(messageId);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (row.player_id === Number(context.playerId)) {
|
|
|
|
|
|
|
|
await bdm.deleteMessage(messageId);
|
|
|
|
|
|
|
|
lobbySocket.emit('messageDelete', messageId);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
|
|
console.error(err);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
socket.on('fieldToggle', async (context, location) => {
|
|
|
|
|
|
|
|
let {row, column} = location;
|
|
|
|
|
|
|
|
let result = await (await (new PlayerWrapper(context.playerId)).grid({lobbyId: lobbyId}))
|
|
|
|
|
|
|
|
.toggleField(row, column);
|
|
|
|
|
|
|
|
socket.emit('fieldChange', await resolveGridField(result));
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
let bdm = new BingoDataManager(pgPool);
|
|
|
|
let bdm = new BingoDataManager(pgPool);
|
|
|
|
|
|
|
|
|
|
|
|
router.init = async () => {
|
|
|
|
router.init = async (bingoIo, io) => {
|
|
|
|
await bdm.init();
|
|
|
|
await bdm.init();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
router.use(async (req, res, next) => {
|
|
|
|
for (let id of await bdm.getLobbyIds())
|
|
|
|
|
|
|
|
createSocketIfNotExist(io, id);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
router.use(async (req, res, next) => {
|
|
|
|
if (req.session.bingoPlayerId)
|
|
|
|
if (req.session.bingoPlayerId)
|
|
|
|
await bdm.updatePlayerExpiration(req.session.bingoPlayerId);
|
|
|
|
await bdm.updatePlayerExpiration(req.session.bingoPlayerId);
|
|
|
|
next();
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
router.get('/', async (req, res) => {
|
|
|
|
router.get('/', async (req, res) => {
|
|
|
|
let playerId = req.session.bingoPlayerId;
|
|
|
|
let playerId = req.session.bingoPlayerId;
|
|
|
|
let info = req.session.acceptedCookies? null: globals.cookieInfo;
|
|
|
|
let info = req.session.acceptedCookies? null: globals.cookieInfo;
|
|
|
|
let lobbyWrapper = new LobbyWrapper(req.query.g);
|
|
|
|
let lobbyWrapper = new LobbyWrapper(req.query.g);
|
|
|
@ -1432,6 +1629,7 @@ router.get('/', async (req, res) => {
|
|
|
|
|
|
|
|
|
|
|
|
if (playerId && await playerWrapper.exists() && req.query.g && await lobbyWrapper.exists()) {
|
|
|
|
if (playerId && await playerWrapper.exists() && req.query.g && await lobbyWrapper.exists()) {
|
|
|
|
let lobbyId = req.query.g;
|
|
|
|
let lobbyId = req.query.g;
|
|
|
|
|
|
|
|
createSocketIfNotExist(io, lobbyId);
|
|
|
|
|
|
|
|
|
|
|
|
if (!(await lobbyWrapper.roundActive() && await playerWrapper.hasGrid(lobbyId))) {
|
|
|
|
if (!(await lobbyWrapper.roundActive() && await playerWrapper.hasGrid(lobbyId))) {
|
|
|
|
if (!await lobbyWrapper.hasPlayer(playerId))
|
|
|
|
if (!await lobbyWrapper.hasPlayer(playerId))
|
|
|
@ -1469,12 +1667,14 @@ router.get('/', async (req, res) => {
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
res.render('bingo/bingo-create', {
|
|
|
|
res.render('bingo/bingo-create', {
|
|
|
|
info: info,
|
|
|
|
info: info,
|
|
|
|
username: await playerWrapper.username()
|
|
|
|
username: await playerWrapper.username(),
|
|
|
|
|
|
|
|
changelog: md.render(globals.changelog),
|
|
|
|
|
|
|
|
primaryJoin: (req.query.g && await lobbyWrapper.exists())
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
router.graphqlResolver = async (req, res) => {
|
|
|
|
router.graphqlResolver = async (req, res) => {
|
|
|
|
let playerId = req.session.bingoPlayerId;
|
|
|
|
let playerId = req.session.bingoPlayerId;
|
|
|
|
if (playerId)
|
|
|
|
if (playerId)
|
|
|
|
await bdm.updatePlayerExpiration(playerId);
|
|
|
|
await bdm.updatePlayerExpiration(playerId);
|
|
|
@ -1496,7 +1696,7 @@ router.graphqlResolver = async (req, res) => {
|
|
|
|
},
|
|
|
|
},
|
|
|
|
// mutations
|
|
|
|
// mutations
|
|
|
|
setUsername: async ({username}) => {
|
|
|
|
setUsername: async ({username}) => {
|
|
|
|
username = replaceTagSigns(username.substring(0, 30)).replace(/[^\w- ;[\]]/g, ''); // only allow 30 characters
|
|
|
|
username = replaceTagSigns(username.substring(0, 30)).replace(/[\n\t👑🌟]|^\s+|\s+$/gu, ''); // only allow 30 characters
|
|
|
|
if (username.length > 0) {
|
|
|
|
if (username.length > 0) {
|
|
|
|
let playerWrapper = new PlayerWrapper(playerId);
|
|
|
|
let playerWrapper = new PlayerWrapper(playerId);
|
|
|
|
|
|
|
|
|
|
|
@ -1506,8 +1706,15 @@ router.graphqlResolver = async (req, res) => {
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
let oldName = await playerWrapper.username();
|
|
|
|
let oldName = await playerWrapper.username();
|
|
|
|
await bdm.updatePlayerUsername(playerId, username);
|
|
|
|
await bdm.updatePlayerUsername(playerId, username);
|
|
|
|
if (req.query.g)
|
|
|
|
|
|
|
|
await bdm.addInfoMessage(req.query.g, `${oldName} changed username to ${username}`);
|
|
|
|
if (req.query.g) {
|
|
|
|
|
|
|
|
let lobbyWrapper = new LobbyWrapper(req.query.g);
|
|
|
|
|
|
|
|
if (await lobbyWrapper.exists()) {
|
|
|
|
|
|
|
|
lobbyWrapper.emit('usernameChange',
|
|
|
|
|
|
|
|
await resolvePlayer(new PlayerWrapper(playerId), req.query.g));
|
|
|
|
|
|
|
|
await lobbyWrapper.addInfoMessage(`${oldName} changed username to ${username}`);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new PlayerWrapper(playerId);
|
|
|
|
return new PlayerWrapper(playerId);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -1519,7 +1726,8 @@ router.graphqlResolver = async (req, res) => {
|
|
|
|
if (playerId)
|
|
|
|
if (playerId)
|
|
|
|
if (gridSize > 0 && gridSize < 10) {
|
|
|
|
if (gridSize > 0 && gridSize < 10) {
|
|
|
|
let result = await bdm.createLobby(playerId, gridSize);
|
|
|
|
let result = await bdm.createLobby(playerId, gridSize);
|
|
|
|
return new LobbyWrapper(result.id);
|
|
|
|
createSocketIfNotExist(io, result.id);
|
|
|
|
|
|
|
|
return new LobbyWrapper(result.id, result);
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
|
res.status(413);
|
|
|
|
res.status(413);
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1527,6 +1735,7 @@ router.graphqlResolver = async (req, res) => {
|
|
|
|
},
|
|
|
|
},
|
|
|
|
mutateLobby: async ({id}) => {
|
|
|
|
mutateLobby: async ({id}) => {
|
|
|
|
let lobbyId = id;
|
|
|
|
let lobbyId = id;
|
|
|
|
|
|
|
|
createSocketIfNotExist(io, lobbyId);
|
|
|
|
await bdm.updateLobbyExpiration(lobbyId);
|
|
|
|
await bdm.updateLobbyExpiration(lobbyId);
|
|
|
|
let lobbyWrapper = new LobbyWrapper(lobbyId);
|
|
|
|
let lobbyWrapper = new LobbyWrapper(lobbyId);
|
|
|
|
return {
|
|
|
|
return {
|
|
|
@ -1639,6 +1848,7 @@ router.graphqlResolver = async (req, res) => {
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
module.exports = router;
|
|
|
|
module.exports = router;
|
|
|
|