mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
fix: Do not overwrite existing root ACL.
Fixes https://github.com/solid/community-server/issues/382
This commit is contained in:
parent
0ecbffa885
commit
77db5c0060
@ -1,10 +1,12 @@
|
|||||||
import type { AclManager } from '../authorization/AclManager';
|
import type { AclManager } from '../authorization/AclManager';
|
||||||
import { RepresentationMetadata } from '../ldp/representation/RepresentationMetadata';
|
import { RepresentationMetadata } from '../ldp/representation/RepresentationMetadata';
|
||||||
|
import type { ResourceIdentifier } from '../ldp/representation/ResourceIdentifier';
|
||||||
import type { LoggerFactory } from '../logging/LoggerFactory';
|
import type { LoggerFactory } from '../logging/LoggerFactory';
|
||||||
import { getLoggerFor, setGlobalLoggerFactory } from '../logging/LogUtil';
|
import { getLoggerFor, setGlobalLoggerFactory } from '../logging/LogUtil';
|
||||||
import type { ExpressHttpServerFactory } from '../server/ExpressHttpServerFactory';
|
import type { HttpServerFactory } from '../server/HttpServerFactory';
|
||||||
import type { ResourceStore } from '../storage/ResourceStore';
|
import type { ResourceStore } from '../storage/ResourceStore';
|
||||||
import { TEXT_TURTLE } from '../util/ContentTypes';
|
import { TEXT_TURTLE } from '../util/ContentTypes';
|
||||||
|
import { NotFoundHttpError } from '../util/errors/NotFoundHttpError';
|
||||||
import { guardedStreamFrom } from '../util/StreamUtil';
|
import { guardedStreamFrom } from '../util/StreamUtil';
|
||||||
import { CONTENT_TYPE } from '../util/UriConstants';
|
import { CONTENT_TYPE } from '../util/UriConstants';
|
||||||
|
|
||||||
@ -13,7 +15,7 @@ import { CONTENT_TYPE } from '../util/UriConstants';
|
|||||||
*/
|
*/
|
||||||
export class Setup {
|
export class Setup {
|
||||||
protected readonly logger = getLoggerFor(this);
|
protected readonly logger = getLoggerFor(this);
|
||||||
private readonly serverFactory: ExpressHttpServerFactory;
|
private readonly serverFactory: HttpServerFactory;
|
||||||
private readonly store: ResourceStore;
|
private readonly store: ResourceStore;
|
||||||
private readonly aclManager: AclManager;
|
private readonly aclManager: AclManager;
|
||||||
private readonly loggerFactory: LoggerFactory;
|
private readonly loggerFactory: LoggerFactory;
|
||||||
@ -21,7 +23,7 @@ export class Setup {
|
|||||||
private readonly port: number;
|
private readonly port: number;
|
||||||
|
|
||||||
public constructor(
|
public constructor(
|
||||||
serverFactory: ExpressHttpServerFactory,
|
serverFactory: HttpServerFactory,
|
||||||
store: ResourceStore,
|
store: ResourceStore,
|
||||||
aclManager: AclManager,
|
aclManager: AclManager,
|
||||||
loggerFactory: LoggerFactory,
|
loggerFactory: LoggerFactory,
|
||||||
@ -40,12 +42,34 @@ export class Setup {
|
|||||||
* Set up a server.
|
* Set up a server.
|
||||||
*/
|
*/
|
||||||
public async setup(): Promise<string> {
|
public async setup(): Promise<string> {
|
||||||
// Configure the logger factory so that others can statically call it.
|
|
||||||
setGlobalLoggerFactory(this.loggerFactory);
|
setGlobalLoggerFactory(this.loggerFactory);
|
||||||
|
|
||||||
// Set up acl so everything can still be done by default
|
const rootAcl = await this.aclManager.getAclDocument({ path: this.base });
|
||||||
|
if (!await this.hasRootAclDocument(rootAcl)) {
|
||||||
|
await this.setRootAclDocument(rootAcl);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.serverFactory.startServer(this.port);
|
||||||
|
return this.base;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async hasRootAclDocument(rootAcl: ResourceIdentifier): Promise<boolean> {
|
||||||
|
try {
|
||||||
|
const result = await this.store.getRepresentation(rootAcl, {});
|
||||||
|
this.logger.debug(`Existing root ACL document found at ${rootAcl.path}`);
|
||||||
|
result.data.destroy();
|
||||||
|
return true;
|
||||||
|
} catch (error: unknown) {
|
||||||
|
if (error instanceof NotFoundHttpError) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up ACL so everything can still be done by default
|
||||||
// Note that this will need to be adapted to go through all the correct channels later on
|
// Note that this will need to be adapted to go through all the correct channels later on
|
||||||
const aclSetup = async(): Promise<void> => {
|
protected async setRootAclDocument(rootAcl: ResourceIdentifier): Promise<void> {
|
||||||
const acl = `@prefix acl: <http://www.w3.org/ns/auth/acl#>.
|
const acl = `@prefix acl: <http://www.w3.org/ns/auth/acl#>.
|
||||||
@prefix foaf: <http://xmlns.com/foaf/0.1/>.
|
@prefix foaf: <http://xmlns.com/foaf/0.1/>.
|
||||||
|
|
||||||
@ -59,22 +83,15 @@ export class Setup {
|
|||||||
acl:mode acl:Control;
|
acl:mode acl:Control;
|
||||||
acl:accessTo <${this.base}>;
|
acl:accessTo <${this.base}>;
|
||||||
acl:default <${this.base}>.`;
|
acl:default <${this.base}>.`;
|
||||||
const baseAclId = await this.aclManager.getAclDocument({ path: this.base });
|
const metadata = new RepresentationMetadata(rootAcl.path, { [CONTENT_TYPE]: TEXT_TURTLE });
|
||||||
const metadata = new RepresentationMetadata(baseAclId.path, { [CONTENT_TYPE]: TEXT_TURTLE });
|
this.logger.debug(`Installing root ACL document at ${rootAcl.path}`);
|
||||||
await this.store.setRepresentation(
|
await this.store.setRepresentation(
|
||||||
baseAclId,
|
rootAcl,
|
||||||
{
|
{
|
||||||
binary: true,
|
binary: true,
|
||||||
data: guardedStreamFrom([ acl ]),
|
data: guardedStreamFrom([ acl ]),
|
||||||
metadata,
|
metadata,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
};
|
|
||||||
this.logger.debug('Setup default ACL settings');
|
|
||||||
await aclSetup();
|
|
||||||
|
|
||||||
this.serverFactory.startServer(this.port);
|
|
||||||
|
|
||||||
return this.base;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,33 +2,61 @@ import type { AclManager } from '../../../src/authorization/AclManager';
|
|||||||
import { Setup } from '../../../src/init/Setup';
|
import { Setup } from '../../../src/init/Setup';
|
||||||
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
||||||
import { VoidLoggerFactory } from '../../../src/logging/VoidLoggerFactory';
|
import { VoidLoggerFactory } from '../../../src/logging/VoidLoggerFactory';
|
||||||
|
import type { HttpServerFactory } from '../../../src/server/HttpServerFactory';
|
||||||
|
import type { ResourceStore } from '../../../src/storage/ResourceStore';
|
||||||
|
import { NotFoundHttpError } from '../../../src/util/errors/NotFoundHttpError';
|
||||||
|
|
||||||
describe('Setup', (): void => {
|
describe('Setup', (): void => {
|
||||||
let serverFactory: any;
|
const serverFactory: jest.Mocked<HttpServerFactory> = {
|
||||||
let store: any;
|
|
||||||
let aclManager: AclManager;
|
|
||||||
let setup: Setup;
|
|
||||||
beforeEach(async(): Promise<void> => {
|
|
||||||
store = {
|
|
||||||
setRepresentation: jest.fn(async(): Promise<void> => undefined),
|
|
||||||
};
|
|
||||||
aclManager = {
|
|
||||||
getAclDocument: jest.fn(async(): Promise<ResourceIdentifier> => ({ path: 'http://test.com/.acl' })),
|
|
||||||
} as any;
|
|
||||||
serverFactory = {
|
|
||||||
startServer: jest.fn(),
|
startServer: jest.fn(),
|
||||||
};
|
};
|
||||||
|
const store: jest.Mocked<ResourceStore> = {
|
||||||
|
getRepresentation: jest.fn().mockRejectedValue(new NotFoundHttpError()),
|
||||||
|
setRepresentation: jest.fn(),
|
||||||
|
} as any;
|
||||||
|
const aclManager: jest.Mocked<AclManager> = {
|
||||||
|
getAclDocument: jest.fn(async(): Promise<ResourceIdentifier> => ({ path: 'http://test.com/.acl' })),
|
||||||
|
} as any;
|
||||||
|
|
||||||
|
let setup: Setup;
|
||||||
|
beforeEach(async(): Promise<void> => {
|
||||||
setup = new Setup(serverFactory, store, aclManager, new VoidLoggerFactory(), 'http://localhost:3000/', 3000);
|
setup = new Setup(serverFactory, store, aclManager, new VoidLoggerFactory(), 'http://localhost:3000/', 3000);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
afterEach((): void => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
});
|
||||||
|
|
||||||
it('starts an HTTP server.', async(): Promise<void> => {
|
it('starts an HTTP server.', async(): Promise<void> => {
|
||||||
await setup.setup();
|
await setup.setup();
|
||||||
|
|
||||||
expect(serverFactory.startServer).toHaveBeenCalledWith(3000);
|
expect(serverFactory.startServer).toHaveBeenCalledWith(3000);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('invokes ACL initialization.', async(): Promise<void> => {
|
it('invokes ACL initialization.', async(): Promise<void> => {
|
||||||
await setup.setup();
|
await setup.setup();
|
||||||
|
|
||||||
expect(aclManager.getAclDocument).toHaveBeenCalledWith({ path: 'http://localhost:3000/' });
|
expect(aclManager.getAclDocument).toHaveBeenCalledWith({ path: 'http://localhost:3000/' });
|
||||||
|
expect(store.getRepresentation).toHaveBeenCalledTimes(1);
|
||||||
|
expect(store.getRepresentation).toHaveBeenCalledWith({ path: 'http://test.com/.acl' }, {});
|
||||||
expect(store.setRepresentation).toHaveBeenCalledTimes(1);
|
expect(store.setRepresentation).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('does not invoke ACL initialization when a root ACL already exists.', async(): Promise<void> => {
|
||||||
|
store.getRepresentation.mockReturnValueOnce(Promise.resolve({
|
||||||
|
data: { destroy: jest.fn() },
|
||||||
|
} as any));
|
||||||
|
|
||||||
|
await setup.setup();
|
||||||
|
|
||||||
|
expect(aclManager.getAclDocument).toHaveBeenCalledWith({ path: 'http://localhost:3000/' });
|
||||||
|
expect(store.getRepresentation).toHaveBeenCalledTimes(1);
|
||||||
|
expect(store.getRepresentation).toHaveBeenCalledWith({ path: 'http://test.com/.acl' }, {});
|
||||||
|
expect(store.setRepresentation).toHaveBeenCalledTimes(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('errors when the root ACL check errors.', async(): Promise<void> => {
|
||||||
|
store.getRepresentation.mockRejectedValueOnce(new Error('Fatal'));
|
||||||
|
await expect(setup.setup()).rejects.toThrow('Fatal');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
Loading…
x
Reference in New Issue
Block a user