Skip to content

Commit df7931c

Browse files
authored
Fix relative file paths with fragments (#603)
* Fix file paths with fragments * Support Windows path in test
1 parent cb5b7e9 commit df7931c

File tree

2 files changed

+37
-0
lines changed

2 files changed

+37
-0
lines changed

src/languageservice/services/yamlSchemaService.ts

+12
Original file line numberDiff line numberDiff line change
@@ -328,12 +328,24 @@ export class YAMLSchemaService extends JSONSchemaService {
328328
let schemaFromModeline = getSchemaFromModeline(doc);
329329
if (schemaFromModeline !== undefined) {
330330
if (!schemaFromModeline.startsWith('file:') && !schemaFromModeline.startsWith('http')) {
331+
// If path contains a fragment and it is left intact, "#" will be
332+
// considered part of the filename and converted to "%23" by
333+
// path.resolve() -> take it out and add back after path.resolve
334+
let appendix = '';
335+
if (schemaFromModeline.indexOf('#') > 0) {
336+
const segments = schemaFromModeline.split('#', 2);
337+
schemaFromModeline = segments[0];
338+
appendix = segments[1];
339+
}
331340
if (!path.isAbsolute(schemaFromModeline)) {
332341
const resUri = URI.parse(resource);
333342
schemaFromModeline = URI.file(path.resolve(path.parse(resUri.fsPath).dir, schemaFromModeline)).toString();
334343
} else {
335344
schemaFromModeline = URI.file(schemaFromModeline).toString();
336345
}
346+
if (appendix.length > 0) {
347+
schemaFromModeline += '#' + appendix;
348+
}
337349
}
338350
this.addSchemaPriority(schemaFromModeline, SchemaPriority.Modeline);
339351
schemas.push(schemaFromModeline);

test/yamlSchemaService.test.ts

+25
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,31 @@ describe('YAML Schema Service', () => {
9292
expect(schema.schema.definitions.bar.type).eqls('string');
9393
});
9494

95+
it('should handle file path with fragments', async () => {
96+
const content = `# yaml-language-server: $schema=schema.json#/definitions/schemaArray\nfoo: bar`;
97+
const yamlDock = parse(content);
98+
99+
requestServiceMock = sandbox.fake.resolves(`{"definitions": {"schemaArray": {
100+
"type": "array",
101+
"minItems": 1,
102+
"items": { "$ref": "#" }
103+
}}, "properties": {}}`);
104+
105+
const service = new SchemaService.YAMLSchemaService(requestServiceMock);
106+
const schema = await service.getSchemaForResource('', yamlDock.documents[0]);
107+
108+
expect(requestServiceMock).calledTwice;
109+
if (process.platform === 'win32') {
110+
expect(requestServiceMock).calledWithExactly('file:///d%3A/schema.json');
111+
expect(requestServiceMock).calledWithExactly('file:///d%3A/schema.json#/definitions/schemaArray');
112+
} else {
113+
expect(requestServiceMock).calledWithExactly('file:///schema.json');
114+
expect(requestServiceMock).calledWithExactly('file:///schema.json#/definitions/schemaArray');
115+
}
116+
117+
expect(schema.schema.type).eqls('array');
118+
});
119+
95120
it('should handle modeline schema comment in the middle of file', () => {
96121
const documentContent = `foo:\n bar\n# yaml-language-server: $schema=https://json-schema.org/draft-07/schema#\naa:bbb\n`;
97122
const content = `${documentContent}`;

0 commit comments

Comments
 (0)