
- Complete NestJS TypeScript implementation with WebSocket support - Direct messaging (DM) and group chat functionality - End-to-end encryption with AES encryption and key pairs - Media file support (images, videos, audio, documents) up to 100MB - Push notifications with Firebase Cloud Messaging integration - Mention alerts and real-time typing indicators - User authentication with JWT and Passport - SQLite database with TypeORM entities and relationships - Comprehensive API documentation with Swagger/OpenAPI - File upload handling with secure access control - Online/offline status tracking and presence management - Message editing, deletion, and reply functionality - Notification management with automatic cleanup - Health check endpoint for monitoring - CORS configuration for cross-origin requests - Environment-based configuration management - Structured for Flutter SDK integration Features implemented: ✅ Real-time messaging with Socket.IO ✅ User registration and authentication ✅ Direct messages and group chats ✅ Media file uploads and management ✅ End-to-end encryption ✅ Push notifications ✅ Mention alerts ✅ Typing indicators ✅ Message read receipts ✅ Online status tracking ✅ File access control ✅ Comprehensive API documentation Ready for Flutter SDK development and production deployment.
96 lines
2.2 KiB
JavaScript
96 lines
2.2 KiB
JavaScript
const memoizer = require('./..');
|
|
const sinon = require('sinon');
|
|
|
|
describe('lru-memoizer sync (events)', function () {
|
|
let memoized;
|
|
let onMiss, onHit, onQueue;
|
|
|
|
beforeEach(function () {
|
|
loadTimes = 0;
|
|
onMiss = sinon.stub();
|
|
onHit = sinon.stub();
|
|
onQueue = sinon.stub();
|
|
memoized = memoizer.sync({
|
|
load: function (a, b, bypass) {
|
|
return a + b;
|
|
},
|
|
hash: function (a, b, bypass) {
|
|
return a + '-' + b;
|
|
},
|
|
bypass: function(a, b, bypass) {
|
|
return bypass;
|
|
},
|
|
max: 10
|
|
});
|
|
memoized.on('hit', onHit);
|
|
memoized.on('miss', onMiss);
|
|
memoized.on('queue', onQueue);
|
|
});
|
|
|
|
describe('when the result is not in the cache', () => {
|
|
beforeEach(() => {
|
|
memoized(1, 2, false);
|
|
});
|
|
|
|
it('should not call onHit', () => {
|
|
sinon.assert.notCalled(onHit);
|
|
});
|
|
|
|
it('should not call onQueue', () => {
|
|
sinon.assert.notCalled(onQueue);
|
|
});
|
|
|
|
it('should call onMiss with the load arguments', () => {
|
|
sinon.assert.calledOnce(onMiss);
|
|
sinon.assert.calledWith(onMiss, 1, 2, false);
|
|
});
|
|
});
|
|
|
|
describe('when the result is in the cache', () => {
|
|
beforeEach(() => {
|
|
memoized(1,2, false);
|
|
onHit.reset();
|
|
onMiss.reset();
|
|
onQueue.reset();
|
|
memoized(1, 2, false);
|
|
});
|
|
|
|
it('should call onHit with the load arguments', () => {
|
|
sinon.assert.calledOnce(onHit);
|
|
sinon.assert.calledWith(onHit, 1, 2, false);
|
|
});
|
|
|
|
it('should not call onQueue', () => {
|
|
sinon.assert.notCalled(onQueue);
|
|
});
|
|
|
|
it('should not call onMiss', () => {
|
|
sinon.assert.notCalled(onQueue);
|
|
});
|
|
});
|
|
|
|
describe('when the cache is by passed', () => {
|
|
beforeEach(() => {
|
|
memoized(1,2, false);
|
|
onHit.reset();
|
|
onMiss.reset();
|
|
onQueue.reset();
|
|
memoized(1, 2, true);
|
|
});
|
|
|
|
it('should not call onHit', () => {
|
|
sinon.assert.notCalled(onHit);
|
|
});
|
|
|
|
it('should not call onQueue', () => {
|
|
sinon.assert.notCalled(onQueue);
|
|
});
|
|
|
|
it('should call onMiss with the load arguments', () => {
|
|
sinon.assert.calledOnce(onMiss);
|
|
sinon.assert.calledWith(onMiss, 1, 2, true);
|
|
});
|
|
});
|
|
});
|
|
|