/* eslint-env node */ const path = require('path'); const { compose } = require('lodash/fp'); const { escapeRegExp, isRegExp, isString } = require('lodash'); const APP_ROOT_PATH = path.join(__dirname, '..', '..', '..'); const SESSION_ID_PATTERN = /\b((05)?[0-9a-f]{64})\b/gi; const SNODE_PATTERN = /(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)/; const GROUP_ID_PATTERN = /(group\()([^)]+)(\))/g; const SERVER_URL_PATTERN = /https?:\/\/[-a-zA-Z0-9@:%._+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_+.~#?&//=]*)/g; const REDACTION_PLACEHOLDER = '[REDACTED]'; // _redactPath :: Path -> String -> String exports._redactPath = filePath => { if (!filePath) { throw new TypeError("'filePath' must be a string"); } const filePathPattern = exports._pathToRegExp(filePath); return text => { if (!isString(text)) { throw new TypeError("'text' must be a string"); } if (!isRegExp(filePathPattern)) { return text; } return text.replace(filePathPattern, REDACTION_PLACEHOLDER); }; }; // _pathToRegExp :: Path -> Maybe RegExp exports._pathToRegExp = filePath => { try { const pathWithNormalizedSlashes = filePath.replace(/\//g, '\\'); const pathWithEscapedSlashes = filePath.replace(/\\/g, '\\\\'); const urlEncodedPath = encodeURI(filePath); // Safe `String::replaceAll`: // https://github.com/lodash/lodash/issues/1084#issuecomment-86698786 const patternString = [ filePath, pathWithNormalizedSlashes, pathWithEscapedSlashes, urlEncodedPath, ] .map(escapeRegExp) .join('|'); return new RegExp(patternString, 'g'); } catch (error) { return null; } }; // Public API // redactSessionID :: String -> String exports.redactSessionID = text => { if (!isString(text)) { throw new TypeError("'text' must be a string"); } return text.replace(SESSION_ID_PATTERN, REDACTION_PLACEHOLDER); }; exports.redactSnodeIP = text => { if (!isString(text)) { throw new TypeError("'text' must be a string"); } return text.replace(SNODE_PATTERN, REDACTION_PLACEHOLDER); }; exports.redactServerUrl = text => { if (!isString(text)) { throw new TypeError("'text' must be a string"); } return text.replace(SERVER_URL_PATTERN, REDACTION_PLACEHOLDER); }; // redactGroupIds :: String -> String exports.redactGroupIds = text => { if (!isString(text)) { throw new TypeError("'text' must be a string"); } return text.replace( GROUP_ID_PATTERN, (match, before, id, after) => `${before}${REDACTION_PLACEHOLDER}${removeNewlines(id).slice(-3)}${after}` ); }; // redactSensitivePaths :: String -> String exports.redactSensitivePaths = exports._redactPath(APP_ROOT_PATH); // redactAll :: String -> String exports.redactAll = compose( exports.redactSensitivePaths, exports.redactGroupIds, exports.redactSessionID, exports.redactSnodeIP, exports.redactServerUrl ); const removeNewlines = text => text.replace(/\r?\n|\r/g, '');