|
|
@ -1,5 +1,5 @@
|
|
|
|
/* jshint esversion: 6 */
|
|
|
|
/* jshint esversion: 6 */
|
|
|
|
/* global Buffer, process */
|
|
|
|
/* global Buffer */
|
|
|
|
;(function () { 'use strict';
|
|
|
|
;(function () { 'use strict';
|
|
|
|
|
|
|
|
|
|
|
|
const nThen = require('nthen');
|
|
|
|
const nThen = require('nthen');
|
|
|
@ -7,9 +7,11 @@ const Nacl = require('tweetnacl');
|
|
|
|
const Crypto = require('crypto');
|
|
|
|
const Crypto = require('crypto');
|
|
|
|
const Once = require("./lib/once");
|
|
|
|
const Once = require("./lib/once");
|
|
|
|
const Meta = require("./lib/metadata");
|
|
|
|
const Meta = require("./lib/metadata");
|
|
|
|
|
|
|
|
const WriteQueue = require("./lib/write-queue");
|
|
|
|
|
|
|
|
|
|
|
|
let Log;
|
|
|
|
let Log;
|
|
|
|
const now = function () { return (new Date()).getTime(); };
|
|
|
|
const now = function () { return (new Date()).getTime(); };
|
|
|
|
|
|
|
|
const ONE_DAY = 1000 * 60 * 60 * 24; // one day in milliseconds
|
|
|
|
|
|
|
|
|
|
|
|
/* getHash
|
|
|
|
/* getHash
|
|
|
|
* this function slices off the leading portion of a message which is
|
|
|
|
* this function slices off the leading portion of a message which is
|
|
|
@ -80,6 +82,7 @@ module.exports.create = function (cfg) {
|
|
|
|
const rpc = cfg.rpc;
|
|
|
|
const rpc = cfg.rpc;
|
|
|
|
const tasks = cfg.tasks;
|
|
|
|
const tasks = cfg.tasks;
|
|
|
|
const store = cfg.store;
|
|
|
|
const store = cfg.store;
|
|
|
|
|
|
|
|
const retainData = cfg.retainData;
|
|
|
|
Log = cfg.log;
|
|
|
|
Log = cfg.log;
|
|
|
|
|
|
|
|
|
|
|
|
Log.silly('HK_LOADING', 'LOADING HISTORY_KEEPER MODULE');
|
|
|
|
Log.silly('HK_LOADING', 'LOADING HISTORY_KEEPER MODULE');
|
|
|
@ -302,20 +305,13 @@ module.exports.create = function (cfg) {
|
|
|
|
* the fix is to use callbacks and implement queueing for writes
|
|
|
|
* the fix is to use callbacks and implement queueing for writes
|
|
|
|
* to guarantee that offset computation is always atomic with writes
|
|
|
|
* to guarantee that offset computation is always atomic with writes
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
const storageQueues = {};
|
|
|
|
const queueStorage = WriteQueue();
|
|
|
|
|
|
|
|
|
|
|
|
const storeQueuedMessage = function (ctx, queue, id) {
|
|
|
|
const storeMessage = function (ctx, channel, msg, isCp, optionalMessageHash) {
|
|
|
|
if (queue.length === 0) {
|
|
|
|
const id = channel.id;
|
|
|
|
delete storageQueues[id];
|
|
|
|
const msgBin = new Buffer(msg + '\n', 'utf8');
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const first = queue.shift();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const msgBin = first.msg;
|
|
|
|
|
|
|
|
const optionalMessageHash = first.hash;
|
|
|
|
|
|
|
|
const isCp = first.isCp;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
queueStorage(id, function (next) {
|
|
|
|
// Store the message first, and update the index only once it's stored.
|
|
|
|
// Store the message first, and update the index only once it's stored.
|
|
|
|
// store.messageBin can be async so updating the index first may
|
|
|
|
// store.messageBin can be async so updating the index first may
|
|
|
|
// result in a wrong cpIndex
|
|
|
|
// result in a wrong cpIndex
|
|
|
@ -331,8 +327,7 @@ module.exports.create = function (cfg) {
|
|
|
|
|
|
|
|
|
|
|
|
// TODO make it possible to respond to clients with errors so they know
|
|
|
|
// TODO make it possible to respond to clients with errors so they know
|
|
|
|
// their message wasn't stored
|
|
|
|
// their message wasn't stored
|
|
|
|
storeQueuedMessage(ctx, queue, id);
|
|
|
|
return void next();
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
}));
|
|
|
|
}).nThen((waitFor) => {
|
|
|
|
}).nThen((waitFor) => {
|
|
|
@ -340,10 +335,7 @@ module.exports.create = function (cfg) {
|
|
|
|
if (err) {
|
|
|
|
if (err) {
|
|
|
|
Log.warn("HK_STORE_MESSAGE_INDEX", err.stack);
|
|
|
|
Log.warn("HK_STORE_MESSAGE_INDEX", err.stack);
|
|
|
|
// non-critical, we'll be able to get the channel index later
|
|
|
|
// non-critical, we'll be able to get the channel index later
|
|
|
|
|
|
|
|
return void next();
|
|
|
|
// proceed to the next message in the queue
|
|
|
|
|
|
|
|
storeQueuedMessage(ctx, queue, id);
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (typeof (index.line) === "number") { index.line++; }
|
|
|
|
if (typeof (index.line) === "number") { index.line++; }
|
|
|
|
if (isCp) {
|
|
|
|
if (isCp) {
|
|
|
@ -362,28 +354,86 @@ module.exports.create = function (cfg) {
|
|
|
|
index.size += msgBin.length;
|
|
|
|
index.size += msgBin.length;
|
|
|
|
|
|
|
|
|
|
|
|
// handle the next element in the queue
|
|
|
|
// handle the next element in the queue
|
|
|
|
storeQueuedMessage(ctx, queue, id);
|
|
|
|
next();
|
|
|
|
}));
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const storeMessage = function (ctx, channel, msg, isCp, optionalMessageHash) {
|
|
|
|
/* historyKeeperBroadcast
|
|
|
|
const id = channel.id;
|
|
|
|
* uses API from the netflux server to send messages to every member of a channel
|
|
|
|
|
|
|
|
* sendMsg runs in a try-catch and drops users if sending a message fails
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
const historyKeeperBroadcast = function (ctx, channel, msg) {
|
|
|
|
|
|
|
|
let chan = ctx.channels[channel] || (([] /*:any*/) /*:Chan_t*/);
|
|
|
|
|
|
|
|
chan.forEach(function (user) {
|
|
|
|
|
|
|
|
sendMsg(ctx, user, [0, HISTORY_KEEPER_ID, 'MSG', user.id, JSON.stringify(msg)]);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const msgBin = new Buffer(msg + '\n', 'utf8');
|
|
|
|
/* expireChannel is here to clean up channels that should have been removed
|
|
|
|
if (Array.isArray(storageQueues[id])) {
|
|
|
|
but for some reason are still present
|
|
|
|
return void storageQueues[id].push({
|
|
|
|
*/
|
|
|
|
msg: msgBin,
|
|
|
|
const expireChannel = function (ctx, channel) {
|
|
|
|
hash: optionalMessageHash,
|
|
|
|
if (retainData) {
|
|
|
|
isCp: isCp,
|
|
|
|
return void store.archiveChannel(channel, function (err) {
|
|
|
|
|
|
|
|
Log.info("ARCHIVAL_CHANNEL_BY_HISTORY_KEEPER_EXPIRATION", {
|
|
|
|
|
|
|
|
channelId: channel,
|
|
|
|
|
|
|
|
status: err? String(err): "SUCCESS",
|
|
|
|
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const queue = storageQueues[id] = (storageQueues[id] || [{
|
|
|
|
store.removeChannel(channel, function (err) {
|
|
|
|
msg: msgBin,
|
|
|
|
Log.info("DELETION_CHANNEL_BY_HISTORY_KEEPER_EXPIRATION", {
|
|
|
|
hash: optionalMessageHash,
|
|
|
|
channelid: channel,
|
|
|
|
}]);
|
|
|
|
status: err? String(err): "SUCCESS",
|
|
|
|
storeQueuedMessage(ctx, queue, id);
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* checkExpired
|
|
|
|
|
|
|
|
* synchronously returns true or undefined to indicate whether the channel is expired
|
|
|
|
|
|
|
|
* according to its metadata
|
|
|
|
|
|
|
|
* has some side effects:
|
|
|
|
|
|
|
|
* closes the channel via the store.closeChannel API
|
|
|
|
|
|
|
|
* and then broadcasts to all channel members that the channel has expired
|
|
|
|
|
|
|
|
* removes the channel from the netflux-server's in-memory cache
|
|
|
|
|
|
|
|
* removes the channel metadata from history keeper's in-memory cache
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
FIXME the boolean nature of this API should be separated from its side effects
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
const checkExpired = function (ctx, channel) {
|
|
|
|
|
|
|
|
if (!(channel && channel.length === STANDARD_CHANNEL_LENGTH)) { return false; }
|
|
|
|
|
|
|
|
let metadata = metadata_cache[channel];
|
|
|
|
|
|
|
|
if (!(metadata && typeof(metadata.expire) === 'number')) { return false; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// the number of milliseconds ago the channel should have expired
|
|
|
|
|
|
|
|
let pastDue = (+new Date()) - metadata.expire;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// less than zero means that it hasn't expired yet
|
|
|
|
|
|
|
|
if (pastDue < 0) { return false; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// if it should have expired more than a day ago...
|
|
|
|
|
|
|
|
// there may have been a problem with scheduling tasks
|
|
|
|
|
|
|
|
// or the scheduled tasks may not be running
|
|
|
|
|
|
|
|
// so trigger a removal from here
|
|
|
|
|
|
|
|
if (pastDue >= ONE_DAY) { expireChannel(ctx, channel); }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// close the channel
|
|
|
|
|
|
|
|
store.closeChannel(channel, function () {
|
|
|
|
|
|
|
|
historyKeeperBroadcast(ctx, channel, {
|
|
|
|
|
|
|
|
error: 'EEXPIRED',
|
|
|
|
|
|
|
|
channel: channel
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
// remove it from any caches after you've told anyone in the channel
|
|
|
|
|
|
|
|
// that it has expired
|
|
|
|
|
|
|
|
delete ctx.channels[channel];
|
|
|
|
|
|
|
|
delete metadata_cache[channel];
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// return true to indicate that it has expired
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var CHECKPOINT_PATTERN = /^cp\|(([A-Za-z0-9+\/=]+)\|)?/;
|
|
|
|
var CHECKPOINT_PATTERN = /^cp\|(([A-Za-z0-9+\/=]+)\|)?/;
|
|
|
@ -436,12 +486,8 @@ module.exports.create = function (cfg) {
|
|
|
|
|
|
|
|
|
|
|
|
metadata = index.metadata;
|
|
|
|
metadata = index.metadata;
|
|
|
|
|
|
|
|
|
|
|
|
if (metadata.expire && metadata.expire < +new Date()) {
|
|
|
|
// don't write messages to expired channels
|
|
|
|
// don't store message sent to expired channels
|
|
|
|
if (checkExpired(ctx, channel)) { return void w.abort(); }
|
|
|
|
w.abort();
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
// TODO if a channel expired a long time ago but it's still here, remove it
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// if there's no validateKey present skip to the next block
|
|
|
|
// if there's no validateKey present skip to the next block
|
|
|
|
if (!metadata.validateKey) { return; }
|
|
|
|
if (!metadata.validateKey) { return; }
|
|
|
@ -674,26 +720,6 @@ module.exports.create = function (cfg) {
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*::
|
|
|
|
|
|
|
|
type Chan_t = {
|
|
|
|
|
|
|
|
indexOf: (any)=>number,
|
|
|
|
|
|
|
|
id: string,
|
|
|
|
|
|
|
|
lastSavedCp: string,
|
|
|
|
|
|
|
|
forEach: ((any)=>void)=>void,
|
|
|
|
|
|
|
|
push: (any)=>void,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* historyKeeperBroadcast
|
|
|
|
|
|
|
|
* uses API from the netflux server to send messages to every member of a channel
|
|
|
|
|
|
|
|
* sendMsg runs in a try-catch and drops users if sending a message fails
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
const historyKeeperBroadcast = function (ctx, channel, msg) {
|
|
|
|
|
|
|
|
let chan = ctx.channels[channel] || (([] /*:any*/) /*:Chan_t*/);
|
|
|
|
|
|
|
|
chan.forEach(function (user) {
|
|
|
|
|
|
|
|
sendMsg(ctx, user, [0, HISTORY_KEEPER_ID, 'MSG', user.id, JSON.stringify(msg)]);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* onChannelCleared
|
|
|
|
/* onChannelCleared
|
|
|
|
* broadcasts to all clients in a channel if that channel is deleted
|
|
|
|
* broadcasts to all clients in a channel if that channel is deleted
|
|
|
@ -729,33 +755,6 @@ module.exports.create = function (cfg) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* checkExpired
|
|
|
|
|
|
|
|
* synchronously returns true or undefined to indicate whether the channel is expired
|
|
|
|
|
|
|
|
* according to its metadata
|
|
|
|
|
|
|
|
* has some side effects:
|
|
|
|
|
|
|
|
* closes the channel via the store.closeChannel API
|
|
|
|
|
|
|
|
* and then broadcasts to all channel members that the channel has expired
|
|
|
|
|
|
|
|
* removes the channel from the netflux-server's in-memory cache
|
|
|
|
|
|
|
|
* removes the channel metadata from history keeper's in-memory cache
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
FIXME the boolean nature of this API should be separated from its side effects
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
const checkExpired = function (ctx, channel) {
|
|
|
|
|
|
|
|
if (channel && channel.length === STANDARD_CHANNEL_LENGTH && metadata_cache[channel] &&
|
|
|
|
|
|
|
|
metadata_cache[channel].expire && metadata_cache[channel].expire < +new Date()) {
|
|
|
|
|
|
|
|
store.closeChannel(channel, function () {
|
|
|
|
|
|
|
|
historyKeeperBroadcast(ctx, channel, {
|
|
|
|
|
|
|
|
error: 'EEXPIRED',
|
|
|
|
|
|
|
|
channel: channel
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
delete ctx.channels[channel];
|
|
|
|
|
|
|
|
delete metadata_cache[channel];
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* onDirectMessage
|
|
|
|
/* onDirectMessage
|
|
|
|
* exported for use by the netflux-server
|
|
|
|
* exported for use by the netflux-server
|
|
|
|
* parses and handles all direct messages directed to the history keeper
|
|
|
|
* parses and handles all direct messages directed to the history keeper
|
|
|
@ -772,7 +771,6 @@ module.exports.create = function (cfg) {
|
|
|
|
const onDirectMessage = function (ctx, seq, user, json) {
|
|
|
|
const onDirectMessage = function (ctx, seq, user, json) {
|
|
|
|
let parsed;
|
|
|
|
let parsed;
|
|
|
|
let channelName;
|
|
|
|
let channelName;
|
|
|
|
let obj = HISTORY_KEEPER_ID;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Log.silly('HK_MESSAGE', json);
|
|
|
|
Log.silly('HK_MESSAGE', json);
|
|
|
|
|
|
|
|
|
|
|
@ -809,6 +807,7 @@ module.exports.create = function (cfg) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
metadata.channel = channelName;
|
|
|
|
metadata.channel = channelName;
|
|
|
|
|
|
|
|
metadata.created = +new Date();
|
|
|
|
|
|
|
|
|
|
|
|
// if the user sends us an invalid key, we won't be able to validate their messages
|
|
|
|
// if the user sends us an invalid key, we won't be able to validate their messages
|
|
|
|
// so they'll never get written to the log anyway. Let's just drop their message
|
|
|
|
// so they'll never get written to the log anyway. Let's just drop their message
|
|
|
@ -913,7 +912,7 @@ module.exports.create = function (cfg) {
|
|
|
|
channelName = parsed[1];
|
|
|
|
channelName = parsed[1];
|
|
|
|
var map = parsed[2];
|
|
|
|
var map = parsed[2];
|
|
|
|
if (!(map && typeof(map) === 'object')) {
|
|
|
|
if (!(map && typeof(map) === 'object')) {
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'INVALID_ARGS', obj]);
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'INVALID_ARGS', HISTORY_KEEPER_ID]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
var oldestKnownHash = map.from;
|
|
|
|
var oldestKnownHash = map.from;
|
|
|
@ -921,11 +920,11 @@ module.exports.create = function (cfg) {
|
|
|
|
var desiredCheckpoint = map.cpCount;
|
|
|
|
var desiredCheckpoint = map.cpCount;
|
|
|
|
var txid = map.txid;
|
|
|
|
var txid = map.txid;
|
|
|
|
if (typeof(desiredMessages) !== 'number' && typeof(desiredCheckpoint) !== 'number') {
|
|
|
|
if (typeof(desiredMessages) !== 'number' && typeof(desiredCheckpoint) !== 'number') {
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'UNSPECIFIED_COUNT', obj]);
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'UNSPECIFIED_COUNT', HISTORY_KEEPER_ID]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!txid) {
|
|
|
|
if (!txid) {
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'NO_TXID', obj]);
|
|
|
|
return void sendMsg(ctx, user, [seq, 'ERROR', 'NO_TXID', HISTORY_KEEPER_ID]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sendMsg(ctx, user, [seq, 'ACK']);
|
|
|
|
sendMsg(ctx, user, [seq, 'ACK']);
|
|
|
@ -1024,33 +1023,6 @@ module.exports.create = function (cfg) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var cciLock = false;
|
|
|
|
|
|
|
|
const checkChannelIntegrity = function (ctx) {
|
|
|
|
|
|
|
|
if (process.env['CRYPTPAD_DEBUG'] && !cciLock) {
|
|
|
|
|
|
|
|
let nt = nThen;
|
|
|
|
|
|
|
|
cciLock = true;
|
|
|
|
|
|
|
|
Object.keys(ctx.channels).forEach(function (channelName) {
|
|
|
|
|
|
|
|
const chan = ctx.channels[channelName];
|
|
|
|
|
|
|
|
if (!chan.index) { return; }
|
|
|
|
|
|
|
|
nt = nt((waitFor) => {
|
|
|
|
|
|
|
|
store.getChannelSize(channelName, waitFor((err, size) => {
|
|
|
|
|
|
|
|
if (err) {
|
|
|
|
|
|
|
|
return void Log.debug("HK_CHECK_CHANNEL_INTEGRITY",
|
|
|
|
|
|
|
|
"Couldn't get size of channel " + channelName);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size !== chan.index.size) {
|
|
|
|
|
|
|
|
return void Log.debug("HK_CHECK_CHANNEL_SIZE",
|
|
|
|
|
|
|
|
"channel size mismatch for " + channelName +
|
|
|
|
|
|
|
|
" --- cached: " + chan.index.size +
|
|
|
|
|
|
|
|
" --- fileSize: " + size);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
}).nThen;
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
nt(() => { cciLock = false; });
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
return {
|
|
|
|
id: HISTORY_KEEPER_ID,
|
|
|
|
id: HISTORY_KEEPER_ID,
|
|
|
|
setConfig: setConfig,
|
|
|
|
setConfig: setConfig,
|
|
|
@ -1058,7 +1030,6 @@ module.exports.create = function (cfg) {
|
|
|
|
dropChannel: dropChannel,
|
|
|
|
dropChannel: dropChannel,
|
|
|
|
checkExpired: checkExpired,
|
|
|
|
checkExpired: checkExpired,
|
|
|
|
onDirectMessage: onDirectMessage,
|
|
|
|
onDirectMessage: onDirectMessage,
|
|
|
|
checkChannelIntegrity: checkChannelIntegrity
|
|
|
|
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|