feat: Implement --baseUrl flag.

Closes https://github.com/solid/community-server/issues/372
This commit is contained in:
Ruben Verborgh 2020-12-01 15:52:44 +01:00
parent 528688bc4c
commit eabe6bc4ed
13 changed files with 60 additions and 66 deletions

View File

@ -6,7 +6,7 @@
"@type": "Variable" "@type": "Variable"
}, },
{ {
"@id": "urn:solid-server:default:variable:base", "@id": "urn:solid-server:default:variable:baseUrl",
"@type": "Variable" "@type": "Variable"
}, },
{ {

View File

@ -31,8 +31,8 @@
}, },
{ {
"@type": "WebSocketAdvertiser", "@type": "WebSocketAdvertiser",
"WebSocketAdvertiser:_settings_port": { "WebSocketAdvertiser:_baseUrl": {
"@id": "urn:solid-server:default:variable:port" "@id": "urn:solid-server:default:variable:baseUrl"
} }
} }
] ]

View File

@ -17,7 +17,7 @@
"@id": "urn:solid-server:default:LoggerFactory" "@id": "urn:solid-server:default:LoggerFactory"
}, },
"Setup:_base": { "Setup:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
}, },
"Setup:_port": { "Setup:_port": {
"@id": "urn:solid-server:default:variable:port" "@id": "urn:solid-server:default:variable:port"

View File

@ -5,7 +5,7 @@
"@id": "urn:solid-server:default:FileIdentifierMapper", "@id": "urn:solid-server:default:FileIdentifierMapper",
"@type": "ExtensionBasedMapper", "@type": "ExtensionBasedMapper",
"ExtensionBasedMapper:_base": { "ExtensionBasedMapper:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
}, },
"ExtensionBasedMapper:_rootFilepath": { "ExtensionBasedMapper:_rootFilepath": {
"@id": "urn:solid-server:default:variable:rootFilePath" "@id": "urn:solid-server:default:variable:rootFilePath"
@ -27,7 +27,7 @@
"@id": "urn:solid-server:default:FileDataAccessor" "@id": "urn:solid-server:default:FileDataAccessor"
}, },
"DataAccessorBasedStore:_base": { "DataAccessorBasedStore:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
} }
] ]

View File

@ -5,7 +5,7 @@
"@id": "urn:solid-server:default:MemoryDataAccessor", "@id": "urn:solid-server:default:MemoryDataAccessor",
"@type": "InMemoryDataAccessor", "@type": "InMemoryDataAccessor",
"InMemoryDataAccessor:_base": { "InMemoryDataAccessor:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
}, },
{ {
@ -15,7 +15,7 @@
"@id": "urn:solid-server:default:MemoryDataAccessor" "@id": "urn:solid-server:default:MemoryDataAccessor"
}, },
"DataAccessorBasedStore:_base": { "DataAccessorBasedStore:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
} }
] ]

View File

@ -8,7 +8,7 @@
"@id": "urn:solid-server:default:variable:sparqlEndpoint" "@id": "urn:solid-server:default:variable:sparqlEndpoint"
}, },
"SparqlDataAccessor:_base": { "SparqlDataAccessor:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
}, },
@ -19,7 +19,7 @@
"@id": "urn:solid-server:default:SparqlDataAccessor" "@id": "urn:solid-server:default:SparqlDataAccessor"
}, },
"DataAccessorBasedStore:_base": { "DataAccessorBasedStore:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
}, },

View File

@ -5,7 +5,7 @@
"@id": "urn:solid-server:default:RegexRouterRule", "@id": "urn:solid-server:default:RegexRouterRule",
"@type": "RegexRouterRule", "@type": "RegexRouterRule",
"RegexRouterRule:_base": { "RegexRouterRule:_base": {
"@id": "urn:solid-server:default:variable:base" "@id": "urn:solid-server:default:variable:baseUrl"
} }
}, },
{ {

View File

@ -4,6 +4,7 @@ import type { LoaderProperties } from 'componentsjs';
import { Loader } from 'componentsjs'; import { Loader } from 'componentsjs';
import yargs from 'yargs'; import yargs from 'yargs';
import { getLoggerFor } from '../logging/LogUtil'; import { getLoggerFor } from '../logging/LogUtil';
import { ensureTrailingSlash } from '../util/PathUtil';
import type { Setup } from './Setup'; import type { Setup } from './Setup';
const logger = getLoggerFor('CliRunner'); const logger = getLoggerFor('CliRunner');
@ -30,11 +31,12 @@ export const runCli = function({
const { argv: params } = yargs(argv.slice(2)) const { argv: params } = yargs(argv.slice(2))
.usage('node ./bin/server.js [args]') .usage('node ./bin/server.js [args]')
.options({ .options({
port: { type: 'number', alias: 'p', default: 3000 }, baseUrl: { type: 'string', alias: 'b' },
config: { type: 'string', alias: 'c' }, config: { type: 'string', alias: 'c' },
loggingLevel: { type: 'string', alias: 'l', default: 'info' },
port: { type: 'number', alias: 'p', default: 3000 },
rootFilePath: { type: 'string', alias: 'f' }, rootFilePath: { type: 'string', alias: 'f' },
sparqlEndpoint: { type: 'string', alias: 's' }, sparqlEndpoint: { type: 'string', alias: 's' },
loggingLevel: { type: 'string', alias: 'l', default: 'info' },
}) })
.help(); .help();
@ -50,16 +52,17 @@ export const runCli = function({
const setup: Setup = await loader const setup: Setup = await loader
.instantiateFromUrl('urn:solid-server:default', configPath, undefined, { .instantiateFromUrl('urn:solid-server:default', configPath, undefined, {
variables: { variables: {
'urn:solid-server:default:variable:baseUrl':
params.baseUrl ? ensureTrailingSlash(params.baseUrl) : `http://localhost:${params.port}/`,
'urn:solid-server:default:variable:loggingLevel': params.loggingLevel,
'urn:solid-server:default:variable:port': params.port, 'urn:solid-server:default:variable:port': params.port,
'urn:solid-server:default:variable:base': `http://localhost:${params.port}/`,
'urn:solid-server:default:variable:rootFilePath': params.rootFilePath ?? process.cwd(), 'urn:solid-server:default:variable:rootFilePath': params.rootFilePath ?? process.cwd(),
'urn:solid-server:default:variable:sparqlEndpoint': params.sparqlEndpoint, 'urn:solid-server:default:variable:sparqlEndpoint': params.sparqlEndpoint,
'urn:solid-server:default:variable:loggingLevel': params.loggingLevel,
}, },
}) as Setup; }) as Setup;
return await setup.setup(); return await setup.setup();
})().then((base: string): void => { })().then((baseUrl: string): void => {
logger.info(`Running at ${base}`); logger.info(`Running at ${baseUrl}`);
}).catch((error): void => { }).catch((error): void => {
// This is the only time we can *not* use the logger to print error messages, as dependency injection has failed. // This is the only time we can *not* use the logger to print error messages, as dependency injection has failed.
stderr.write(`${error}\n`); stderr.write(`${error}\n`);

View File

@ -2,26 +2,16 @@ import { addHeader } from '../../util/HeaderUtil';
import { HttpHandler } from '../HttpHandler'; import { HttpHandler } from '../HttpHandler';
import type { HttpResponse } from '../HttpResponse'; import type { HttpResponse } from '../HttpResponse';
interface WebSocketSettings {
hostname?: string;
port?: number;
protocol?: string;
}
/** /**
* Handler that advertises a WebSocket through the Updates-Via header. * Handler that advertises a WebSocket through the Updates-Via header.
*/ */
export class WebSocketAdvertiser extends HttpHandler { export class WebSocketAdvertiser extends HttpHandler {
private readonly socketUrl: string; private readonly socketUrl: string;
public constructor(settings: WebSocketSettings = {}) { public constructor(baseUrl: string) {
super(); super();
const { hostname = 'localhost', port = 80, protocol = 'ws:' } = settings; const socketUrl = new URL(baseUrl);
const secure = /^(?:https|wss)/u.test(protocol); socketUrl.protocol = /^(?:http|ws):/u.test(baseUrl) ? 'ws:' : 'wss:';
const socketUrl = new URL(`${secure ? 'wss' : 'ws'}://${hostname}:${port}/`);
if (socketUrl.hostname !== hostname) {
throw new Error(`Invalid hostname: ${hostname}`);
}
this.socketUrl = socketUrl.href; this.socketUrl = socketUrl.href;
} }

View File

@ -23,6 +23,7 @@ describe('An Express server with middleware', (): void => {
'urn:solid-server:default:ExpressHttpServerFactory', 'middleware.json', { 'urn:solid-server:default:ExpressHttpServerFactory', 'middleware.json', {
'urn:solid-server:default:LdpHandler': new SimpleHttpHandler(), 'urn:solid-server:default:LdpHandler': new SimpleHttpHandler(),
'urn:solid-server:default:variable:port': port, 'urn:solid-server:default:variable:port': port,
'urn:solid-server:default:variable:baseUrl': 'https://example.pod/',
}, },
) as ExpressHttpServerFactory; ) as ExpressHttpServerFactory;
server = factory.startServer(port); server = factory.startServer(port);

View File

@ -5,7 +5,7 @@ import type { HttpServerFactory } from '../../src/server/HttpServerFactory';
import { instantiateFromConfig } from '../configs/Util'; import { instantiateFromConfig } from '../configs/Util';
const port = 6001; const port = 6001;
const baseUrl = `http://localhost:${port}/`; const serverUrl = `http://localhost:${port}/`;
describe('A server with the Solid WebSockets API', (): void => { describe('A server with the Solid WebSockets API', (): void => {
let server: Server; let server: Server;
@ -14,7 +14,7 @@ describe('A server with the Solid WebSockets API', (): void => {
const factory = await instantiateFromConfig( const factory = await instantiateFromConfig(
'urn:solid-server:default:ServerFactory', 'websockets.json', { 'urn:solid-server:default:ServerFactory', 'websockets.json', {
'urn:solid-server:default:variable:port': port, 'urn:solid-server:default:variable:port': port,
'urn:solid-server:default:variable:base': baseUrl, 'urn:solid-server:default:variable:baseUrl': 'http://example.pod/',
}, },
) as HttpServerFactory; ) as HttpServerFactory;
server = factory.startServer(port); server = factory.startServer(port);
@ -27,17 +27,17 @@ describe('A server with the Solid WebSockets API', (): void => {
}); });
it('returns a 200.', async(): Promise<void> => { it('returns a 200.', async(): Promise<void> => {
const response = await fetch(baseUrl); const response = await fetch(serverUrl, { headers: { host: 'example.pod' }});
expect(response.status).toBe(200); expect(response.status).toBe(200);
}); });
it('sets the Updates-Via header.', async(): Promise<void> => { it('sets the Updates-Via header.', async(): Promise<void> => {
const response = await fetch(baseUrl); const response = await fetch(serverUrl, { headers: { host: 'example.pod' }});
expect(response.headers.get('Updates-Via')).toBe(`ws://localhost:${port}/`); expect(response.headers.get('Updates-Via')).toBe('ws://example.pod/');
}); });
it('exposes the Updates-Via header via CORS.', async(): Promise<void> => { it('exposes the Updates-Via header via CORS.', async(): Promise<void> => {
const response = await fetch(baseUrl); const response = await fetch(serverUrl, { headers: { host: 'example.pod' }});
expect(response.headers.get('Access-Control-Expose-Headers')!.split(',')) expect(response.headers.get('Access-Control-Expose-Headers')!.split(','))
.toContain('Updates-Via'); .toContain('Updates-Via');
}); });
@ -47,7 +47,7 @@ describe('A server with the Solid WebSockets API', (): void => {
const messages = new Array<string>(); const messages = new Array<string>();
beforeAll(async(): Promise<void> => { beforeAll(async(): Promise<void> => {
client = new WebSocket(`ws://localhost:${port}`, [ 'solid/0.1.0-alpha' ]); client = new WebSocket(`ws://localhost:${port}`, [ 'solid/0.1.0-alpha' ], { headers: { host: 'example.pod' }});
client.on('message', (message: string): any => messages.push(message)); client.on('message', (message: string): any => messages.push(message));
await new Promise((resolve): any => client.on('open', resolve)); await new Promise((resolve): any => client.on('open', resolve));
}); });
@ -69,21 +69,24 @@ describe('A server with the Solid WebSockets API', (): void => {
describe('when the client subscribes to a resource', (): void => { describe('when the client subscribes to a resource', (): void => {
beforeAll(async(): Promise<void> => { beforeAll(async(): Promise<void> => {
client.send(`sub ${baseUrl}my-resource`); client.send(`sub http://example.pod/my-resource`);
await new Promise((resolve): any => client.once('message', resolve)); await new Promise((resolve): any => client.once('message', resolve));
}); });
it('acknowledges the subscription.', async(): Promise<void> => { it('acknowledges the subscription.', async(): Promise<void> => {
expect(messages).toEqual([ `ack ${baseUrl}my-resource` ]); expect(messages).toEqual([ `ack http://example.pod/my-resource` ]);
}); });
it('notifies the client of resource updates.', async(): Promise<void> => { it('notifies the client of resource updates.', async(): Promise<void> => {
await fetch(`${baseUrl}my-resource`, { await fetch(`${serverUrl}my-resource`, {
method: 'PUT', method: 'PUT',
headers: { 'content-type': 'application/json' }, headers: {
host: 'example.pod',
'content-type': 'application/json',
},
body: '{}', body: '{}',
}); });
expect(messages).toEqual([ `pub ${baseUrl}my-resource` ]); expect(messages).toEqual([ `pub http://example.pod/my-resource` ]);
}); });
}); });
}); });

View File

@ -5,13 +5,13 @@ import type { Setup } from '../../../src/init/Setup';
const mainModulePath = path.join(__dirname, '../../../'); const mainModulePath = path.join(__dirname, '../../../');
const mockSetup = { const mockSetup: jest.Mocked<Setup> = {
setup: jest.fn(async(): Promise<any> => null), setup: jest.fn(async(): Promise<any> => null),
} as unknown as jest.Mocked<Setup>; } as any;
const loader = { const loader: jest.Mocked<Loader> = {
instantiateFromUrl: jest.fn(async(): Promise<any> => mockSetup), instantiateFromUrl: jest.fn(async(): Promise<any> => mockSetup),
registerAvailableModuleResources: jest.fn(async(): Promise<any> => mockSetup), registerAvailableModuleResources: jest.fn(async(): Promise<any> => mockSetup),
} as unknown as jest.Mocked<Loader>; } as any;
// Mock the Loader class. // Mock the Loader class.
jest.mock('componentsjs', (): any => ({ jest.mock('componentsjs', (): any => ({
@ -40,7 +40,7 @@ describe('CliRunner', (): void => {
{ {
variables: { variables: {
'urn:solid-server:default:variable:port': 3000, 'urn:solid-server:default:variable:port': 3000,
'urn:solid-server:default:variable:base': `http://localhost:3000/`, 'urn:solid-server:default:variable:baseUrl': 'http://localhost:3000/',
'urn:solid-server:default:variable:rootFilePath': process.cwd(), 'urn:solid-server:default:variable:rootFilePath': process.cwd(),
'urn:solid-server:default:variable:sparqlEndpoint': undefined, 'urn:solid-server:default:variable:sparqlEndpoint': undefined,
'urn:solid-server:default:variable:loggingLevel': 'info', 'urn:solid-server:default:variable:loggingLevel': 'info',
@ -58,6 +58,7 @@ describe('CliRunner', (): void => {
argv: [ argv: [
'node', 'script', 'node', 'script',
'-p', '4000', '-p', '4000',
'-b', 'http://pod.example/',
'-c', 'myconfig.json', '-c', 'myconfig.json',
'-f', '/root', '-f', '/root',
'-s', 'http://localhost:5000/sparql', '-s', 'http://localhost:5000/sparql',
@ -73,7 +74,7 @@ describe('CliRunner', (): void => {
{ {
variables: { variables: {
'urn:solid-server:default:variable:port': 4000, 'urn:solid-server:default:variable:port': 4000,
'urn:solid-server:default:variable:base': `http://localhost:4000/`, 'urn:solid-server:default:variable:baseUrl': 'http://pod.example/',
'urn:solid-server:default:variable:rootFilePath': '/root', 'urn:solid-server:default:variable:rootFilePath': '/root',
'urn:solid-server:default:variable:sparqlEndpoint': 'http://localhost:5000/sparql', 'urn:solid-server:default:variable:sparqlEndpoint': 'http://localhost:5000/sparql',
'urn:solid-server:default:variable:loggingLevel': 'debug', 'urn:solid-server:default:variable:loggingLevel': 'debug',
@ -87,6 +88,7 @@ describe('CliRunner', (): void => {
argv: [ argv: [
'node', 'script', 'node', 'script',
'--port', '4000', '--port', '4000',
'--baseUrl', 'http://pod.example/',
'--config', 'myconfig.json', '--config', 'myconfig.json',
'--rootFilePath', '/root', '--rootFilePath', '/root',
'--sparqlEndpoint', 'http://localhost:5000/sparql', '--sparqlEndpoint', 'http://localhost:5000/sparql',
@ -102,7 +104,7 @@ describe('CliRunner', (): void => {
{ {
variables: { variables: {
'urn:solid-server:default:variable:port': 4000, 'urn:solid-server:default:variable:port': 4000,
'urn:solid-server:default:variable:base': `http://localhost:4000/`, 'urn:solid-server:default:variable:baseUrl': 'http://pod.example/',
'urn:solid-server:default:variable:rootFilePath': '/root', 'urn:solid-server:default:variable:rootFilePath': '/root',
'urn:solid-server:default:variable:sparqlEndpoint': 'http://localhost:5000/sparql', 'urn:solid-server:default:variable:sparqlEndpoint': 'http://localhost:5000/sparql',
'urn:solid-server:default:variable:loggingLevel': 'debug', 'urn:solid-server:default:variable:loggingLevel': 'debug',

View File

@ -2,36 +2,31 @@ import { createResponse } from 'node-mocks-http';
import { WebSocketAdvertiser } from '../../../../src/server/middleware/WebSocketAdvertiser'; import { WebSocketAdvertiser } from '../../../../src/server/middleware/WebSocketAdvertiser';
describe('A WebSocketAdvertiser', (): void => { describe('A WebSocketAdvertiser', (): void => {
it('writes a default HTTP WebSocket.', async(): Promise<void> => { it('writes a ws: socket when given an http: URL.', async(): Promise<void> => {
const writer = new WebSocketAdvertiser(); const writer = new WebSocketAdvertiser('http://test.example/');
const response = createResponse();
await writer.handle({ response } as any);
expect(response.getHeaders()).toEqual({ 'updates-via': 'ws://localhost/' });
});
it('writes an HTTP WebSocket with port 80.', async(): Promise<void> => {
const writer = new WebSocketAdvertiser({ hostname: 'test.example', port: 80, protocol: 'http' });
const response = createResponse(); const response = createResponse();
await writer.handle({ response } as any); await writer.handle({ response } as any);
expect(response.getHeaders()).toEqual({ 'updates-via': 'ws://test.example/' }); expect(response.getHeaders()).toEqual({ 'updates-via': 'ws://test.example/' });
}); });
it('writes an HTTP WebSocket with port 3000.', async(): Promise<void> => { it('writes a ws: socket when given a ws: URL.', async(): Promise<void> => {
const writer = new WebSocketAdvertiser({ hostname: 'test.example', port: 3000, protocol: 'http' }); const writer = new WebSocketAdvertiser('ws://test.example/');
const response = createResponse(); const response = createResponse();
await writer.handle({ response } as any); await writer.handle({ response } as any);
expect(response.getHeaders()).toEqual({ 'updates-via': 'ws://test.example:3000/' }); expect(response.getHeaders()).toEqual({ 'updates-via': 'ws://test.example/' });
}); });
it('writes an HTTPS WebSocket with port 443.', async(): Promise<void> => { it('writes a wss: socket when given an https: URL.', async(): Promise<void> => {
const writer = new WebSocketAdvertiser({ hostname: 'test.example', port: 443, protocol: 'https' }); const writer = new WebSocketAdvertiser('https://test.example/');
const response = createResponse(); const response = createResponse();
await writer.handle({ response } as any); await writer.handle({ response } as any);
expect(response.getHeaders()).toEqual({ 'updates-via': 'wss://test.example/' }); expect(response.getHeaders()).toEqual({ 'updates-via': 'wss://test.example/' });
}); });
it('rejects an invalid hostname.', (): void => { it('writes a wss: socket when given a wss: URL.', async(): Promise<void> => {
expect((): any => new WebSocketAdvertiser({ hostname: 'test.example/invalid' })) const writer = new WebSocketAdvertiser('wss://test.example/');
.toThrow('Invalid hostname: test.example/invalid'); const response = createResponse();
await writer.handle({ response } as any);
expect(response.getHeaders()).toEqual({ 'updates-via': 'wss://test.example/' });
}); });
}); });