mirror of
https://github.com/compiler-explorer/compiler-explorer.git
synced 2025-12-27 14:04:04 -05:00
212 lines
7.1 KiB
TypeScript
212 lines
7.1 KiB
TypeScript
// Copyright (c) 2022, Compiler Explorer Authors
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright notice,
|
|
// this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above copyright
|
|
// notice, this list of conditions and the following disclaimer in the
|
|
// documentation and/or other materials provided with the distribution.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
export function isString(x: any): x is string {
|
|
return typeof x === 'string' || x instanceof String;
|
|
}
|
|
|
|
// Object.keys is typed as returning :string[] for some reason
|
|
// This util is for cases where the key is a union of a few possible keys and we
|
|
// want the resulting array properly typed.
|
|
export function keys<K extends string | number | symbol>(o: Partial<Record<K, any>>): K[] {
|
|
return Object.keys(o) as K[];
|
|
}
|
|
|
|
export function unique<V>(arr: V[]): V[] {
|
|
return [...new Set(arr)];
|
|
}
|
|
|
|
export function intersection<V>(a: V[], b: V[]): V[] {
|
|
const B = new Set(b);
|
|
return [...a].filter(item => B.has(item));
|
|
}
|
|
|
|
// arr.filter(x => x !== null) returns a (T | null)[] even though it is a T[]
|
|
// Apparently the idiomatic solution is arr.filter((x): x is T => x !== null), but this is shorter (and the type
|
|
// predicate also isn't type checked so it doesn't seem safe to me)
|
|
export function remove<U, V extends U>(arr: U[], v: V) {
|
|
return arr.filter(item => item !== v) as Exclude<U, V extends null | undefined ? V : never>[];
|
|
}
|
|
|
|
// https://www.typescriptlang.org/play?#code/KYDwDg9gTgLgBAMwK4DsDGMCWEVysAWwgDdgAeAVQBo4A1OUGYFAEwGc4KA+ACgEMoUAFycA2gF0axEbQCUcAN4AoOKrzAYSKLgFQAdAkwAbJlB6YmBOAF4ucC4TgBCa9bjF5fDgFEQaI0gs5NR0DCBMrBwoSEZGcAA+cKhBhijALHAA-KEiaaRQXBIA3EoAvkpKQf4CwHBoOGzwuiI8jVCYKADmCXDRsbLFFfUojXAgNupEpPyCNH1GsiUA9EtqcAB6mUMN8ACeE-hTwDNQNABECBAQZ4tKK2ubFUA
|
|
|
|
// For Array.prototype.sort
|
|
export function basic_comparator<T>(a: T, b: T) {
|
|
if (a < b) {
|
|
return -1;
|
|
}
|
|
if (a > b) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// https://stackoverflow.com/questions/41253310/typescript-retrieve-element-type-information-from-array-type
|
|
export type ElementType<ArrayType extends readonly unknown[]> = ArrayType extends readonly (infer T)[] ? T : never;
|
|
|
|
const EscapeMap = {
|
|
'&': '&',
|
|
'<': '<',
|
|
'>': '>',
|
|
'"': '"',
|
|
"'": ''',
|
|
'`': '`',
|
|
};
|
|
const EscapeRE = new RegExp(`(?:${Object.keys(EscapeMap).join('|')})`, 'g');
|
|
|
|
export function escapeHTML(text: string) {
|
|
return text.replace(EscapeRE, str => EscapeMap[str as keyof typeof EscapeMap]);
|
|
}
|
|
|
|
function splitIntoChunks(s: string, chunkSize: number): string[] {
|
|
const chunks: string[] = [];
|
|
const isNegative = s.slice(0, 1) === '-';
|
|
if (isNegative) {
|
|
s = s.slice(1);
|
|
}
|
|
const firstChunkLength = s.length % chunkSize;
|
|
if (firstChunkLength !== 0) {
|
|
chunks.push(s.slice(0, firstChunkLength));
|
|
}
|
|
for (let i = firstChunkLength; i < s.length; i += chunkSize) {
|
|
chunks.push(s.slice(i, i + chunkSize));
|
|
}
|
|
if (isNegative) {
|
|
chunks[0] = '-' + (chunks[0] ?? '');
|
|
}
|
|
return chunks;
|
|
}
|
|
|
|
export function addDigitSeparator(n: string, digitSeparator: string, chunkSize: number): string {
|
|
return splitIntoChunks(n, chunkSize).join(digitSeparator);
|
|
}
|
|
|
|
class ArgumentParser {
|
|
private rest: string;
|
|
private position: number;
|
|
private inDoubleQuotes: boolean;
|
|
private readonly results: string[];
|
|
|
|
constructor(str: string) {
|
|
this.rest = str.trim();
|
|
this.position = 0;
|
|
this.inDoubleQuotes = false;
|
|
this.results = [];
|
|
}
|
|
|
|
private addResult(result: string) {
|
|
if (this.results[this.position] === undefined) {
|
|
this.results[this.position] = result;
|
|
} else {
|
|
this.results[this.position] += result;
|
|
}
|
|
}
|
|
|
|
private handleEscapeCharacter() {
|
|
if (this.rest.length < 2) {
|
|
this.addResult('\\');
|
|
} else {
|
|
if (this.rest.charAt(1) !== '\n') {
|
|
this.addResult(this.rest.charAt(1));
|
|
}
|
|
}
|
|
this.rest = this.rest.slice(2);
|
|
}
|
|
|
|
private handleSingleQuotes() {
|
|
let quotePos = this.rest.indexOf("'", 1);
|
|
if (quotePos === -1) quotePos = this.rest.length;
|
|
this.addResult(this.rest.slice(1, quotePos));
|
|
this.rest = this.rest.slice(quotePos + 1);
|
|
}
|
|
|
|
private handleDoubleQuotes() {
|
|
const match = /["\\]/.exec(this.rest);
|
|
const matchEnd = match ? match.index : this.rest.length;
|
|
|
|
this.addResult(this.rest.slice(0, matchEnd));
|
|
|
|
if (match && match[0] === '\\') {
|
|
this.rest = this.rest.slice(matchEnd);
|
|
this.handleEscapeCharacter();
|
|
} else {
|
|
this.rest = this.rest.slice(matchEnd + 1);
|
|
this.inDoubleQuotes = false;
|
|
}
|
|
}
|
|
|
|
private handleWhitespace() {
|
|
const wsMatch = /^\s+/.exec(this.rest);
|
|
if (wsMatch) {
|
|
this.rest = this.rest.slice(wsMatch[0].length);
|
|
++this.position;
|
|
}
|
|
}
|
|
|
|
private handleNonDoubleQuote(nextMatch: RegExpExecArray) {
|
|
this.addResult(this.rest.slice(0, nextMatch.index));
|
|
this.rest = this.rest.slice(nextMatch.index);
|
|
|
|
switch (nextMatch[0]) {
|
|
case '\\':
|
|
this.handleEscapeCharacter();
|
|
break;
|
|
case "'":
|
|
this.handleSingleQuotes();
|
|
break;
|
|
case '"':
|
|
this.rest = this.rest.slice(1);
|
|
this.inDoubleQuotes = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
public exec(): string[] {
|
|
while (this.rest) {
|
|
if (this.inDoubleQuotes) {
|
|
this.handleDoubleQuotes();
|
|
continue;
|
|
}
|
|
|
|
this.handleWhitespace();
|
|
|
|
const nextMatch = /[\s'"\\]/.exec(this.rest);
|
|
if (!nextMatch) {
|
|
this.addResult(this.rest);
|
|
this.rest = '';
|
|
continue;
|
|
}
|
|
|
|
this.handleNonDoubleQuote(nextMatch);
|
|
}
|
|
return this.results;
|
|
}
|
|
}
|
|
|
|
export function splitArguments(str = ''): string[] {
|
|
return new ArgumentParser(str).exec();
|
|
}
|