
- 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.
46 lines
1.3 KiB
JavaScript
46 lines
1.3 KiB
JavaScript
'use strict';
|
|
|
|
var punycode = require('punycode/');
|
|
var $encode = punycode.ucs2.encode;
|
|
|
|
var regexTest = require('safe-regex-test');
|
|
var callBound = require('call-bound');
|
|
var $TypeError = require('es-errors/type');
|
|
|
|
var entities = require('./entities.json');
|
|
|
|
var endsInSemicolon = regexTest(/;$/);
|
|
|
|
var $replace = callBound('String.prototype.replace');
|
|
var $exec = callBound('RegExp.prototype.exec');
|
|
var $parseInt = parseInt;
|
|
|
|
module.exports = function decode(str) {
|
|
if (typeof str !== 'string') {
|
|
throw new $TypeError('Expected a String');
|
|
}
|
|
|
|
return $replace(str, /&(#?[^;\W]+;?)/g, function (_, match) {
|
|
var m = $exec(/^#(\d+);?$/, match);
|
|
if (m) {
|
|
return $encode([$parseInt(m[1], 10)]);
|
|
}
|
|
var m2 = $exec(/^#[Xx]([A-Fa-f0-9]+);?/, match);
|
|
if (m2) {
|
|
return $encode([$parseInt(m2[1], 16)]);
|
|
}
|
|
// named entity
|
|
var hasSemi = endsInSemicolon(match);
|
|
var withoutSemi = hasSemi ? $replace(match, /;$/, '') : match;
|
|
var target = entities[withoutSemi] || (hasSemi && entities[match]);
|
|
|
|
if (typeof target === 'number') {
|
|
return $encode([target]);
|
|
} else if (typeof target === 'string') {
|
|
return target;
|
|
}
|
|
return '&' + match;
|
|
|
|
});
|
|
};
|