Não pode escolher mais do que 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.
 

358 linhas
11 KiB

'use strict';
import { createHash, HexBase64Latin1Encoding } from 'crypto';
const emptyStr = '';
export namespace Strings {
export const enum CharCode {
/**
* The `/` character.
*/
Slash = 47,
/**
* The `\` character.
*/
Backslash = 92
}
const escapeMarkdownRegex = /[\\`*_{}[\]()#+\-.!]/g;
const escapeMarkdownHeaderRegex = /^===/gm;
// const sampleMarkdown = '## message `not code` *not important* _no underline_ \n> don\'t quote me \n- don\'t list me \n+ don\'t list me \n1. don\'t list me \nnot h1 \n=== \nnot h2 \n---\n***\n---\n___';
const markdownQuotedRegex = /\n/g;
export function escapeMarkdown(s: string, options: { quoted?: boolean } = {}): string {
s = s
// Escape markdown
.replace(escapeMarkdownRegex, '\\$&')
// Escape markdown header (since the above regex won't match it)
.replace(escapeMarkdownHeaderRegex, '\u200b===');
if (!options.quoted) return s;
// Keep under the same block-quote but with line breaks
return s.replace(markdownQuotedRegex, '\t\n> ');
}
export function getCommonBase(s1: string, s2: string, delimiter: string) {
let char;
let index = 0;
for (let i = 0; i < s1.length; i++) {
char = s1[i];
if (char !== s2[i]) break;
if (char === delimiter) {
index = i;
}
}
return index > 0 ? s1.substring(0, index + 1) : undefined;
}
export function getDurationMilliseconds(start: [number, number]) {
const [secs, nanosecs] = process.hrtime(start);
return secs * 1000 + Math.floor(nanosecs / 1000000);
}
const pathNormalizeRegex = /\\/g;
const pathStripTrailingSlashRegex = /\/$/g;
const tokenRegex = /\$\{(\W*)?([^|]*?)(?:\|(\d+)(-|\?)?)?(\W*)?\}/g;
const tokenSanitizeRegex = /\$\{(?:\W*)?(\w*?)(?:[\W\d]*)\}/g;
// eslint-disable-next-line no-template-curly-in-string
const tokenSanitizeReplacement = '$${this.$1}';
export interface TokenOptions {
collapseWhitespace: boolean;
padDirection: 'left' | 'right';
prefix: string | undefined;
suffix: string | undefined;
truncateTo: number | undefined;
}
export function getTokensFromTemplate(template: string) {
const tokens: { key: string; options: TokenOptions }[] = [];
let match;
do {
match = tokenRegex.exec(template);
if (match == null) break;
const [, prefix, key, truncateTo, option, suffix] = match;
tokens.push({
key: key,
options: {
collapseWhitespace: option === '?',
padDirection: option === '-' ? 'left' : 'right',
prefix: prefix,
suffix: suffix,
truncateTo: truncateTo == null ? undefined : parseInt(truncateTo, 10)
}
});
} while (match != null);
return tokens;
}
const interpolationMap = new Map<string, Function>();
export function interpolate(template: string, context: object | undefined): string {
if (template == null || template.length === 0) return template;
if (context === undefined) return template.replace(tokenSanitizeRegex, emptyStr);
let fn = interpolationMap.get(template);
if (fn === undefined) {
fn = new Function(`return \`${template.replace(tokenSanitizeRegex, tokenSanitizeReplacement)}\`;`);
interpolationMap.set(template, fn);
}
return fn.call(context);
}
export function* lines(s: string): IterableIterator<string> {
let i = 0;
while (i < s.length) {
let j = s.indexOf('\n', i);
if (j === -1) {
j = s.length;
}
yield s.substring(i, j);
i = j + 1;
}
}
export function md5(s: string, encoding: HexBase64Latin1Encoding = 'base64'): string {
return createHash('md5')
.update(s)
.digest(encoding);
}
export function normalizePath(
fileName: string,
options: { addLeadingSlash?: boolean; stripTrailingSlash?: boolean } = { stripTrailingSlash: true }
) {
if (fileName == null || fileName.length === 0) return fileName;
let normalized = fileName.replace(pathNormalizeRegex, '/');
const { addLeadingSlash, stripTrailingSlash } = { stripTrailingSlash: true, ...options };
if (stripTrailingSlash) {
normalized = normalized.replace(pathStripTrailingSlashRegex, emptyStr);
}
if (addLeadingSlash && normalized.charCodeAt(0) !== CharCode.Slash) {
normalized = `/${normalized}`;
}
return normalized;
}
export function pad(s: string, before: number = 0, after: number = 0, padding: string = '\u00a0') {
if (before === 0 && after === 0) return s;
return `${before === 0 ? emptyStr : padding.repeat(before)}${s}${
after === 0 ? emptyStr : padding.repeat(after)
}`;
}
export function padLeft(s: string, padTo: number, padding: string = '\u00a0', width?: number) {
const diff = padTo - (width || getWidth(s));
return diff <= 0 ? s : padding.repeat(diff) + s;
}
export function padLeftOrTruncate(s: string, max: number, padding?: string, width?: number) {
width = width || getWidth(s);
if (width < max) return padLeft(s, max, padding, width);
if (width > max) return truncate(s, max, undefined, width);
return s;
}
export function padRight(s: string, padTo: number, padding: string = '\u00a0', width?: number) {
const diff = padTo - (width || getWidth(s));
return diff <= 0 ? s : s + padding.repeat(diff);
}
export function padOrTruncate(s: string, max: number, padding?: string, width?: number) {
const left = max < 0;
max = Math.abs(max);
width = width || getWidth(s);
if (width < max) return left ? padLeft(s, max, padding, width) : padRight(s, max, padding, width);
if (width > max) return truncate(s, max, undefined, width);
return s;
}
export function padRightOrTruncate(s: string, max: number, padding?: string, width?: number) {
width = width || getWidth(s);
if (width < max) return padRight(s, max, padding, width);
if (width > max) return truncate(s, max);
return s;
}
export function pluralize(
s: string,
count: number,
options?: { number?: string; plural?: string; suffix?: string; zero?: string }
) {
if (options === undefined) return `${count} ${s}${count === 1 ? emptyStr : 's'}`;
return `${count === 0 ? options.zero || count : options.number || count} ${
count === 1 ? s : options.plural || `${s}${options.suffix || 's'}`
}`;
}
// Removes \ / : * ? " < > | and C0 and C1 control codes
// eslint-disable-next-line no-control-regex
const illegalCharsForFSRegex = /[\\/:*?"<>|\x00-\x1f\x80-\x9f]/g;
export function sanitizeForFileSystem(s: string, replacement: string = '_') {
if (!s) return s;
return s.replace(illegalCharsForFSRegex, replacement);
}
export function sha1(s: string, encoding: HexBase64Latin1Encoding = 'base64'): string {
return createHash('sha1')
.update(s)
.digest(encoding);
}
export function splitLast(s: string, splitter: string) {
const index = s.lastIndexOf(splitter);
if (index === -1) return [s];
return [s.substr(index), s.substring(0, index - 1)];
}
export function splitSingle(s: string, splitter: string) {
const parts = s.split(splitter, 1);
const first = parts[0];
return first.length === s.length ? parts : [first, s.substr(first.length + 1)];
}
export function truncate(s: string, truncateTo: number, ellipsis: string = '\u2026', width?: number) {
if (!s) return s;
width = width || getWidth(s);
if (width <= truncateTo) return s;
if (width === s.length) return `${s.substring(0, truncateTo - 1)}${ellipsis}`;
// Skip ahead to start as far as we can by assuming all the double-width characters won't be truncated
let chars = Math.floor(truncateTo / (width / s.length));
let count = getWidth(s.substring(0, chars));
while (count < truncateTo) {
count += getWidth(s[chars++]);
}
if (count >= truncateTo) {
chars--;
}
return `${s.substring(0, chars)}${ellipsis}`;
}
const ansiRegex = /[\\u001B\\u009B][[\\]()#;?]*(?:(?:(?:[a-zA-Z\\d]*(?:;[a-zA-Z\\d]*)*)?\\u0007)|(?:(?:\\d{1,4}(?:;\\d{0,4})*)?[\\dA-PRZcf-ntqry=><~]))/g;
const containsNonAsciiRegex = /[^\x20-\x7F\u00a0\u2026]/;
export function getWidth(s: string): number {
if (s == null || s.length === 0) return 0;
// Shortcut to avoid needless string `RegExp`s, replacements, and allocations
if (!containsNonAsciiRegex.test(s)) return s.length;
s = s.replace(ansiRegex, emptyStr);
let count = 0;
let emoji = 0;
let joiners = 0;
const graphemes = [...s];
for (let i = 0; i < graphemes.length; i++) {
const code = graphemes[i].codePointAt(0)!;
// Ignore control characters
if (code <= 0x1f || (code >= 0x7f && code <= 0x9f)) continue;
// Ignore combining characters
if (code >= 0x300 && code <= 0x36f) continue;
// https://stackoverflow.com/questions/30757193/find-out-if-character-in-string-is-emoji
if (
(code >= 0x1f600 && code <= 0x1f64f) || // Emoticons
(code >= 0x1f300 && code <= 0x1f5ff) || // Misc Symbols and Pictographs
(code >= 0x1f680 && code <= 0x1f6ff) || // Transport and Map
(code >= 0x2600 && code <= 0x26ff) || // Misc symbols
(code >= 0x2700 && code <= 0x27bf) || // Dingbats
(code >= 0xfe00 && code <= 0xfe0f) || // Variation Selectors
(code >= 0x1f900 && code <= 0x1f9ff) || // Supplemental Symbols and Pictographs
(code >= 65024 && code <= 65039) || // Variation selector
(code >= 8400 && code <= 8447) // Combining Diacritical Marks for Symbols
) {
if (code >= 0x1f3fb && code <= 0x1f3ff) continue; // emoji modifier fitzpatrick type
emoji++;
count += 2;
continue;
}
// Ignore zero-width joiners '\u200d'
if (code === 8205) {
joiners++;
count -= 2;
continue;
}
// Surrogates
if (code > 0xffff) {
i++;
}
count += isFullwidthCodePoint(code) ? 2 : 1;
}
const offset = emoji - joiners;
if (offset > 1) {
count += offset - 1;
}
return count;
}
function isFullwidthCodePoint(cp: number) {
// code points are derived from:
// http://www.unix.org/Public/UNIDATA/EastAsianWidth.txt
if (
cp >= 0x1100 &&
(cp <= 0x115f || // Hangul Jamo
cp === 0x2329 || // LEFT-POINTING ANGLE BRACKET
cp === 0x232a || // RIGHT-POINTING ANGLE BRACKET
// CJK Radicals Supplement .. Enclosed CJK Letters and Months
(cp >= 0x2e80 && cp <= 0x3247 && cp !== 0x303f) ||
// Enclosed CJK Letters and Months .. CJK Unified Ideographs Extension A
(cp >= 0x3250 && cp <= 0x4dbf) ||
// CJK Unified Ideographs .. Yi Radicals
(cp >= 0x4e00 && cp <= 0xa4c6) ||
// Hangul Jamo Extended-A
(cp >= 0xa960 && cp <= 0xa97c) ||
// Hangul Syllables
(cp >= 0xac00 && cp <= 0xd7a3) ||
// CJK Compatibility Ideographs
(cp >= 0xf900 && cp <= 0xfaff) ||
// Vertical Forms
(cp >= 0xfe10 && cp <= 0xfe19) ||
// CJK Compatibility Forms .. Small Form Variants
(cp >= 0xfe30 && cp <= 0xfe6b) ||
// Halfwidth and Fullwidth Forms
(cp >= 0xff01 && cp <= 0xff60) ||
(cp >= 0xffe0 && cp <= 0xffe6) ||
// Kana Supplement
(cp >= 0x1b000 && cp <= 0x1b001) ||
// Enclosed Ideographic Supplement
(cp >= 0x1f200 && cp <= 0x1f251) ||
// CJK Unified Ideographs Extension B .. Tertiary Ideographic Plane
(cp >= 0x20000 && cp <= 0x3fffd))
) {
return true;
}
return false;
}
}