2020-08-17 17:09:56 -04:00
|
|
|
import { languages } from 'monaco-editor';
|
2020-05-22 05:08:09 -04:00
|
|
|
import {
|
|
|
|
isTextFile,
|
|
|
|
registerLanguages,
|
2020-09-16 08:10:15 -04:00
|
|
|
registerSchema,
|
2020-05-22 05:08:09 -04:00
|
|
|
trimPathComponents,
|
2020-05-25 08:08:23 -04:00
|
|
|
insertFinalNewline,
|
|
|
|
trimTrailingWhitespace,
|
2020-05-22 05:08:09 -04:00
|
|
|
getPathParents,
|
2020-06-01 08:08:28 -04:00
|
|
|
getPathParent,
|
|
|
|
readFileAsDataURL,
|
2020-05-22 05:08:09 -04:00
|
|
|
} from '~/ide/utils';
|
2019-06-26 10:50:31 -04:00
|
|
|
|
|
|
|
describe('WebIDE utils', () => {
|
2020-03-11 17:09:19 -04:00
|
|
|
describe('isTextFile', () => {
|
2020-09-17 05:09:32 -04:00
|
|
|
it.each`
|
|
|
|
mimeType | name | type | result
|
|
|
|
${'image/png'} | ${'my.png'} | ${'binary'} | ${false}
|
|
|
|
${'IMAGE/PNG'} | ${'my.png'} | ${'binary'} | ${false}
|
|
|
|
${'text/plain'} | ${'my.txt'} | ${'text'} | ${true}
|
|
|
|
${'TEXT/PLAIN'} | ${'my.txt'} | ${'text'} | ${true}
|
|
|
|
`('returns $result for known $type types', ({ mimeType, name, result }) => {
|
|
|
|
expect(isTextFile({ content: 'file content', mimeType, name })).toBe(result);
|
2020-03-11 17:09:19 -04:00
|
|
|
});
|
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it.each`
|
|
|
|
content | mimeType | name
|
|
|
|
${'{"éêė":"value"}'} | ${'application/json'} | ${'my.json'}
|
|
|
|
${'{"éêė":"value"}'} | ${'application/json'} | ${'.tsconfig'}
|
|
|
|
${'SELECT "éêė" from tablename'} | ${'application/sql'} | ${'my.sql'}
|
|
|
|
${'{"éêė":"value"}'} | ${'application/json'} | ${'MY.JSON'}
|
|
|
|
${'SELECT "éêė" from tablename'} | ${'application/sql'} | ${'MY.SQL'}
|
|
|
|
${'var code = "something"'} | ${'application/javascript'} | ${'Gruntfile'}
|
|
|
|
${'MAINTAINER Александр "a21283@me.com"'} | ${'application/octet-stream'} | ${'dockerfile'}
|
|
|
|
`(
|
|
|
|
'returns true for file extensions that Monaco supports syntax highlighting for',
|
|
|
|
({ content, mimeType, name }) => {
|
|
|
|
expect(isTextFile({ content, mimeType, name })).toBe(true);
|
|
|
|
},
|
|
|
|
);
|
2020-03-11 17:09:19 -04:00
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it('returns false if filename is same as the expected extension', () => {
|
|
|
|
expect(
|
|
|
|
isTextFile({
|
|
|
|
name: 'sql',
|
|
|
|
content: 'SELECT "éêė" from tablename',
|
|
|
|
mimeType: 'application/sql',
|
|
|
|
}),
|
|
|
|
).toBeFalsy();
|
2020-03-11 17:09:19 -04:00
|
|
|
});
|
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it('returns true for ASCII only content for unknown types', () => {
|
2020-03-11 17:09:19 -04:00
|
|
|
expect(
|
2020-09-17 05:09:32 -04:00
|
|
|
isTextFile({
|
|
|
|
name: 'hello.mytype',
|
|
|
|
content: 'plain text',
|
|
|
|
mimeType: 'application/x-new-type',
|
|
|
|
}),
|
2020-03-11 17:09:19 -04:00
|
|
|
).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it('returns false for non-ASCII content for unknown types', () => {
|
|
|
|
expect(
|
|
|
|
isTextFile({
|
|
|
|
name: 'my.random',
|
|
|
|
content: '{"éêė":"value"}',
|
|
|
|
mimeType: 'application/octet-stream',
|
|
|
|
}),
|
|
|
|
).toBeFalsy();
|
2020-03-11 17:09:19 -04:00
|
|
|
});
|
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it.each`
|
|
|
|
name | result
|
|
|
|
${'myfile.txt'} | ${true}
|
|
|
|
${'Dockerfile'} | ${true}
|
|
|
|
${'img.png'} | ${false}
|
|
|
|
${'abc.js'} | ${true}
|
|
|
|
${'abc.random'} | ${false}
|
|
|
|
${'image.jpeg'} | ${false}
|
|
|
|
`('returns $result for $filename when no content or mimeType is passed', ({ name, result }) => {
|
|
|
|
expect(isTextFile({ name })).toBe(result);
|
2020-03-11 17:09:19 -04:00
|
|
|
});
|
|
|
|
|
2020-09-17 05:09:32 -04:00
|
|
|
it('returns true if content is empty string but false if content is not passed', () => {
|
|
|
|
expect(isTextFile({ name: 'abc.dat' })).toBe(false);
|
|
|
|
expect(isTextFile({ name: 'abc.dat', content: '' })).toBe(true);
|
|
|
|
expect(isTextFile({ name: 'abc.dat', content: ' ' })).toBe(true);
|
2020-03-11 17:09:19 -04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-15 05:07:59 -04:00
|
|
|
describe('trimPathComponents', () => {
|
|
|
|
it.each`
|
|
|
|
input | output
|
|
|
|
${'example path '} | ${'example path'}
|
|
|
|
${'p/somefile '} | ${'p/somefile'}
|
|
|
|
${'p /somefile '} | ${'p/somefile'}
|
|
|
|
${'p/ somefile '} | ${'p/somefile'}
|
|
|
|
${' p/somefile '} | ${'p/somefile'}
|
|
|
|
${'p/somefile .md'} | ${'p/somefile .md'}
|
|
|
|
${'path / to / some/file.doc '} | ${'path/to/some/file.doc'}
|
|
|
|
`('trims all path components in path: "$input"', ({ input, output }) => {
|
|
|
|
expect(trimPathComponents(input)).toEqual(output);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-07 11:09:29 -04:00
|
|
|
describe('registerLanguages', () => {
|
|
|
|
let langs;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
langs = [
|
|
|
|
{
|
|
|
|
id: 'html',
|
|
|
|
extensions: ['.html'],
|
|
|
|
conf: { comments: { blockComment: ['<!--', '-->'] } },
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'css',
|
|
|
|
extensions: ['.css'],
|
|
|
|
conf: { comments: { blockComment: ['/*', '*/'] } },
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'js',
|
|
|
|
extensions: ['.js'],
|
|
|
|
conf: { comments: { blockComment: ['/*', '*/'] } },
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
jest.spyOn(languages, 'register').mockImplementation(() => {});
|
|
|
|
jest.spyOn(languages, 'setMonarchTokensProvider').mockImplementation(() => {});
|
|
|
|
jest.spyOn(languages, 'setLanguageConfiguration').mockImplementation(() => {});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('registers all the passed languages with Monaco', () => {
|
|
|
|
registerLanguages(...langs);
|
|
|
|
|
|
|
|
expect(languages.register.mock.calls).toEqual([
|
|
|
|
[
|
|
|
|
{
|
|
|
|
conf: { comments: { blockComment: ['/*', '*/'] } },
|
|
|
|
extensions: ['.css'],
|
|
|
|
id: 'css',
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
],
|
|
|
|
[
|
|
|
|
{
|
|
|
|
conf: { comments: { blockComment: ['/*', '*/'] } },
|
|
|
|
extensions: ['.js'],
|
|
|
|
id: 'js',
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
],
|
|
|
|
[
|
|
|
|
{
|
|
|
|
conf: { comments: { blockComment: ['<!--', '-->'] } },
|
|
|
|
extensions: ['.html'],
|
|
|
|
id: 'html',
|
|
|
|
language: { tokenizer: {} },
|
|
|
|
},
|
|
|
|
],
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(languages.setMonarchTokensProvider.mock.calls).toEqual([
|
|
|
|
['css', { tokenizer: {} }],
|
|
|
|
['js', { tokenizer: {} }],
|
|
|
|
['html', { tokenizer: {} }],
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(languages.setLanguageConfiguration.mock.calls).toEqual([
|
|
|
|
['css', { comments: { blockComment: ['/*', '*/'] } }],
|
|
|
|
['js', { comments: { blockComment: ['/*', '*/'] } }],
|
|
|
|
['html', { comments: { blockComment: ['<!--', '-->'] } }],
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
2020-05-22 05:08:09 -04:00
|
|
|
|
2020-09-16 08:10:15 -04:00
|
|
|
describe('registerSchema', () => {
|
|
|
|
let schema;
|
2020-07-07 11:08:49 -04:00
|
|
|
|
|
|
|
beforeEach(() => {
|
2020-09-16 08:10:15 -04:00
|
|
|
schema = {
|
|
|
|
uri: 'http://myserver/foo-schema.json',
|
|
|
|
fileMatch: ['*'],
|
|
|
|
schema: {
|
|
|
|
id: 'http://myserver/foo-schema.json',
|
|
|
|
type: 'object',
|
|
|
|
properties: {
|
|
|
|
p1: { enum: ['v1', 'v2'] },
|
|
|
|
p2: { $ref: 'http://myserver/bar-schema.json' },
|
2020-07-07 11:08:49 -04:00
|
|
|
},
|
2020-09-16 08:10:15 -04:00
|
|
|
},
|
2020-07-07 11:08:49 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
jest.spyOn(languages.json.jsonDefaults, 'setDiagnosticsOptions');
|
|
|
|
jest.spyOn(languages.yaml.yamlDefaults, 'setDiagnosticsOptions');
|
|
|
|
});
|
|
|
|
|
2020-09-16 08:10:15 -04:00
|
|
|
it('registers the given schemas with monaco for both json and yaml languages', () => {
|
|
|
|
registerSchema(schema);
|
|
|
|
|
|
|
|
expect(languages.json.jsonDefaults.setDiagnosticsOptions).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({ schemas: [schema] }),
|
|
|
|
);
|
|
|
|
expect(languages.yaml.yamlDefaults.setDiagnosticsOptions).toHaveBeenCalledWith(
|
|
|
|
expect.objectContaining({ schemas: [schema] }),
|
|
|
|
);
|
|
|
|
});
|
2020-07-07 11:08:49 -04:00
|
|
|
});
|
|
|
|
|
2020-05-25 08:08:23 -04:00
|
|
|
describe('trimTrailingWhitespace', () => {
|
|
|
|
it.each`
|
|
|
|
input | output
|
|
|
|
${'text \n more text \n'} | ${'text\n more text\n'}
|
|
|
|
${'text \n more text \n\n \n'} | ${'text\n more text\n\n\n'}
|
|
|
|
${'text \t\t \n more text \n\t\ttext\n \n\t\t'} | ${'text\n more text\n\t\ttext\n\n'}
|
|
|
|
${'text \r\n more text \r\n'} | ${'text\r\n more text\r\n'}
|
|
|
|
${'text \r\n more text \r\n\r\n \r\n'} | ${'text\r\n more text\r\n\r\n\r\n'}
|
|
|
|
${'text \t\t \r\n more text \r\n\t\ttext\r\n \r\n\t\t'} | ${'text\r\n more text\r\n\t\ttext\r\n\r\n'}
|
|
|
|
`("trims trailing whitespace in each line of file's contents: $input", ({ input, output }) => {
|
|
|
|
expect(trimTrailingWhitespace(input)).toBe(output);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-22 05:08:09 -04:00
|
|
|
describe('addFinalNewline', () => {
|
|
|
|
it.each`
|
|
|
|
input | output
|
|
|
|
${'some text'} | ${'some text\n'}
|
|
|
|
${'some text\n'} | ${'some text\n'}
|
|
|
|
${'some text\n\n'} | ${'some text\n\n'}
|
|
|
|
${'some\n text'} | ${'some\n text\n'}
|
|
|
|
`('adds a newline if it doesnt already exist for input: $input', ({ input, output }) => {
|
2020-05-25 08:08:23 -04:00
|
|
|
expect(insertFinalNewline(input)).toBe(output);
|
2020-05-22 05:08:09 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
it.each`
|
|
|
|
input | output
|
|
|
|
${'some text'} | ${'some text\r\n'}
|
|
|
|
${'some text\r\n'} | ${'some text\r\n'}
|
|
|
|
${'some text\n'} | ${'some text\n\r\n'}
|
|
|
|
${'some text\r\n\r\n'} | ${'some text\r\n\r\n'}
|
|
|
|
${'some\r\n text'} | ${'some\r\n text\r\n'}
|
|
|
|
`('works with CRLF newline style; input: $input', ({ input, output }) => {
|
2020-05-25 08:08:23 -04:00
|
|
|
expect(insertFinalNewline(input, '\r\n')).toBe(output);
|
2020-05-22 05:08:09 -04:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getPathParents', () => {
|
|
|
|
it.each`
|
|
|
|
path | parents
|
|
|
|
${'foo/bar/baz/index.md'} | ${['foo/bar/baz', 'foo/bar', 'foo']}
|
|
|
|
${'foo/bar/baz'} | ${['foo/bar', 'foo']}
|
|
|
|
${'index.md'} | ${[]}
|
|
|
|
${'path with/spaces to/something.md'} | ${['path with/spaces to', 'path with']}
|
|
|
|
`('gets all parent directory names for path: $path', ({ path, parents }) => {
|
|
|
|
expect(getPathParents(path)).toEqual(parents);
|
|
|
|
});
|
2020-06-01 08:08:28 -04:00
|
|
|
|
|
|
|
it.each`
|
|
|
|
path | depth | parents
|
|
|
|
${'foo/bar/baz/index.md'} | ${0} | ${[]}
|
|
|
|
${'foo/bar/baz/index.md'} | ${1} | ${['foo/bar/baz']}
|
|
|
|
${'foo/bar/baz/index.md'} | ${2} | ${['foo/bar/baz', 'foo/bar']}
|
|
|
|
${'foo/bar/baz/index.md'} | ${3} | ${['foo/bar/baz', 'foo/bar', 'foo']}
|
|
|
|
${'foo/bar/baz/index.md'} | ${4} | ${['foo/bar/baz', 'foo/bar', 'foo']}
|
|
|
|
`('gets only the immediate $depth parents if when depth=$depth', ({ path, depth, parents }) => {
|
|
|
|
expect(getPathParents(path, depth)).toEqual(parents);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getPathParent', () => {
|
|
|
|
it.each`
|
|
|
|
path | parents
|
|
|
|
${'foo/bar/baz/index.md'} | ${'foo/bar/baz'}
|
|
|
|
${'foo/bar/baz'} | ${'foo/bar'}
|
|
|
|
${'index.md'} | ${undefined}
|
|
|
|
${'path with/spaces to/something.md'} | ${'path with/spaces to'}
|
|
|
|
`('gets the immediate parent for path: $path', ({ path, parents }) => {
|
|
|
|
expect(getPathParent(path)).toEqual(parents);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('readFileAsDataURL', () => {
|
|
|
|
it('reads a file and returns its output as a data url', () => {
|
|
|
|
const file = new File(['foo'], 'foo.png', { type: 'image/png' });
|
|
|
|
|
|
|
|
return readFileAsDataURL(file).then(contents => {
|
|
|
|
expect(contents).toBe('');
|
|
|
|
});
|
|
|
|
});
|
2020-05-22 05:08:09 -04:00
|
|
|
});
|
2019-06-26 10:50:31 -04:00
|
|
|
});
|