debian-mirror-gitlab/app/assets/javascripts/ide/utils.js

183 lines
5.1 KiB
JavaScript
Raw Normal View History

2020-11-24 15:15:51 +05:30
import { flatten, isString } from 'lodash';
2021-03-11 19:13:27 +05:30
import { languages } from 'monaco-editor';
2022-11-25 23:54:43 +05:30
import { setDiagnosticsOptions as yamlDiagnosticsOptions } from 'monaco-yaml';
2021-01-29 00:20:46 +05:30
import { performanceMarkAndMeasure } from '~/performance/utils';
2021-03-11 19:13:27 +05:30
import { SIDE_LEFT, SIDE_RIGHT } from './constants';
2020-04-08 14:13:33 +05:30
2021-03-08 18:12:59 +05:30
const toLowerCase = (x) => x.toLowerCase();
2020-04-08 14:13:33 +05:30
const monacoLanguages = languages.getLanguages();
const monacoExtensions = new Set(
2021-03-08 18:12:59 +05:30
flatten(monacoLanguages.map((lang) => lang.extensions?.map(toLowerCase) || [])),
2020-04-08 14:13:33 +05:30
);
const monacoMimetypes = new Set(
2021-03-08 18:12:59 +05:30
flatten(monacoLanguages.map((lang) => lang.mimetypes?.map(toLowerCase) || [])),
2020-04-08 14:13:33 +05:30
);
const monacoFilenames = new Set(
2021-03-08 18:12:59 +05:30
flatten(monacoLanguages.map((lang) => lang.filenames?.map(toLowerCase) || [])),
2020-04-08 14:13:33 +05:30
);
const KNOWN_TYPES = [
{
isText: false,
isMatch(mimeType) {
return mimeType.toLowerCase().includes('image/');
},
},
{
isText: true,
isMatch(mimeType) {
return mimeType.toLowerCase().includes('text/');
},
},
{
isText: true,
isMatch(mimeType, fileName) {
const fileExtension = fileName.includes('.') ? `.${fileName.split('.').pop()}` : '';
return (
monacoExtensions.has(fileExtension.toLowerCase()) ||
monacoMimetypes.has(mimeType.toLowerCase()) ||
monacoFilenames.has(fileName.toLowerCase())
);
},
},
];
2021-11-11 11:23:49 +05:30
export function isTextFile({ name, raw, binary, content, mimeType = '' }) {
// some file objects already have a `binary` property set on them. If so, use it first
if (typeof binary === 'boolean') return !binary;
2021-03-08 18:12:59 +05:30
const knownType = KNOWN_TYPES.find((type) => type.isMatch(mimeType, name));
2020-04-08 14:13:33 +05:30
if (knownType) return knownType.isText;
// does the string contain ascii characters only (ranges from space to tilde, tabs and new lines)
const asciiRegex = /^[ -~\t\n\r]+$/;
2020-11-24 15:15:51 +05:30
2021-01-03 14:25:43 +05:30
const fileContents = raw || content;
2020-04-08 14:13:33 +05:30
// for unknown types, determine the type by evaluating the file contents
2021-01-03 14:25:43 +05:30
return isString(fileContents) && (fileContents === '' || asciiRegex.test(fileContents));
2020-04-08 14:13:33 +05:30
}
2018-11-18 11:00:15 +05:30
2021-03-08 18:12:59 +05:30
export const createPathWithExt = (p) => {
2018-11-18 11:00:15 +05:30
const ext = p.lastIndexOf('.') >= 0 ? p.substring(p.lastIndexOf('.') + 1) : '';
return `${p.substring(1, p.lastIndexOf('.') + 1 || p.length)}${ext || '.js'}`;
};
2020-05-24 23:13:21 +05:30
2021-03-08 18:12:59 +05:30
export const trimPathComponents = (path) =>
2020-05-24 23:13:21 +05:30
path
.split('/')
2021-03-08 18:12:59 +05:30
.map((s) => s.trim())
2020-05-24 23:13:21 +05:30
.join('/');
export function registerLanguages(def, ...defs) {
2021-03-08 18:12:59 +05:30
defs.forEach((lang) => registerLanguages(lang));
2020-05-24 23:13:21 +05:30
const languageId = def.id;
languages.register(def);
languages.setMonarchTokensProvider(languageId, def.language);
languages.setLanguageConfiguration(languageId, def.conf);
}
2020-06-23 00:09:42 +05:30
2022-07-23 23:45:48 +05:30
export function registerSchema(schema, options = {}) {
2022-11-25 23:54:43 +05:30
const defaultOptions = {
validate: true,
enableSchemaRequest: true,
hover: true,
completion: true,
schemas: [schema],
...options,
};
languages.json.jsonDefaults.setDiagnosticsOptions(defaultOptions);
yamlDiagnosticsOptions(defaultOptions);
2020-07-28 23:09:34 +05:30
}
2021-03-08 18:12:59 +05:30
export const otherSide = (side) => (side === SIDE_RIGHT ? SIDE_LEFT : SIDE_RIGHT);
2020-06-23 00:09:42 +05:30
export function trimTrailingWhitespace(content) {
return content.replace(/[^\S\r\n]+$/gm, '');
}
export function getPathParents(path, maxDepth = Infinity) {
const pathComponents = path.split('/');
const paths = [];
let depth = 0;
while (pathComponents.length && depth < maxDepth) {
pathComponents.pop();
let parentPath = pathComponents.join('/');
if (parentPath.startsWith('/')) parentPath = parentPath.slice(1);
if (parentPath) paths.push(parentPath);
depth += 1;
}
return paths;
}
export function getPathParent(path) {
return getPathParents(path, 1)[0];
}
/**
* Takes a file object and returns a data uri of its contents.
*
* @param {File} file
*/
export function readFileAsDataURL(file) {
2021-03-08 18:12:59 +05:30
return new Promise((resolve) => {
2020-06-23 00:09:42 +05:30
const reader = new FileReader();
2021-03-08 18:12:59 +05:30
reader.addEventListener('load', (e) => resolve(e.target.result), { once: true });
2020-06-23 00:09:42 +05:30
reader.readAsDataURL(file);
});
}
export function getFileEOL(content = '') {
return content.includes('\r\n') ? 'CRLF' : 'LF';
}
2021-01-03 14:25:43 +05:30
/**
* Adds or increments the numeric suffix to a filename/branch name.
* Retains underscore or dash before the numeric suffix if it already exists.
*
* Examples:
* hello -> hello-1
* hello-2425 -> hello-2425
* hello.md -> hello-1.md
* hello_2.md -> hello_3.md
* hello_ -> hello_1
2021-09-04 01:27:46 +05:30
* main-patch-22432 -> main-patch-22433
2021-01-03 14:25:43 +05:30
* patch_332 -> patch_333
*
* @param {string} filename File name or branch name
* @param {number} [randomize] Should randomize the numeric suffix instead of auto-incrementing?
*/
export function addNumericSuffix(filename, randomize = false) {
return filename.replace(/([ _-]?)(\d*)(\..+?$|$)/, (_, before, number, after) => {
2021-03-08 18:12:59 +05:30
const n = randomize ? Math.random().toString().substring(2, 7).slice(-5) : Number(number) + 1;
2021-01-03 14:25:43 +05:30
return `${before || '-'}${n}${after}`;
});
}
export const measurePerformance = (
mark,
measureName,
measureStart = undefined,
measureEnd = mark,
) => {
performanceMarkAndMeasure({
mark,
measures: [
{
name: measureName,
start: measureStart,
end: measureEnd,
},
],
});
};