validate Nodeinfo response by schema (#2390)

* rm stable: 'false' from actions/setup-go@v3

* adapt tests from #2369

* set undefined as defaultStreamKey

pass adminpass to sendConfigChangeRequest()

* mv getAdminConfig to api/lib/config.js

* npm install --quiet for automated tests

* refactor tests

separate default values from new ones

* test adminpass change

fix defaultStreamKeys test

* fix defaultStreamKeys

* use getAdminStatus

* mv test/automated/lib/config.js to admin.js

* check default hideViewerCount

cleanup

* test more default options in api

erverName
SServerSummary
yp.instanceUrl
FederationConfig.username

* more testing of default config params

* update reference values for api test
This commit is contained in:
Meisam 2022-11-30 00:49:08 +01:00 committed by Gabe Kangas
parent f4c2a49887
commit 0a8fc6e8c5
8 changed files with 653 additions and 119 deletions

View file

@ -27,7 +27,6 @@ jobs:
- uses: actions/checkout@v3
- uses: actions/setup-go@v3
with:
stable: 'false'
go-version: '1.17.2'
- name: Run HLS tests
run: cd test/automated/hls && ./run.sh

View file

@ -1,19 +1,87 @@
var request = require('supertest');
const Random = require('crypto-random');
const sendConfigChangeRequest = require('./lib/admin').sendConfigChangeRequest;
const getAdminConfig = require('./lib/admin').getAdminConfig;
const getAdminStatus = require('./lib/admin').getAdminStatus;
request = request('http://127.0.0.1:8080');
const serverName = randomString();
const streamTitle = randomString();
const serverSummary = randomString();
const offlineMessage = randomString();
const pageContent = `<p>${randomString()}</p>`;
const tags = [randomString(), randomString(), randomString()];
const streamKeys = [
{ key: randomString(), comment: 'test key 1' },
{ key: randomString(), comment: 'test key 1' },
{ key: randomString(), comment: 'test key 1' },
];
const latencyLevel = Math.floor(Math.random() * 4);
// initial configuration of server
const defaultServerName = 'New Owncast Server';
const defaultStreamTitle = undefined;
const defaultLogo = '/logo';
const defaultOfflineMessage = '';
const defaultServerSummary = 'This is a new live video streaming server powered by Owncast.';
const defaultAdminPassword = 'abc123';
const defaultStreamKeys = [{ key: defaultAdminPassword, comment: 'Default stream key' }];
const defaultTags = ["owncast", "streaming"];
const defaultYPConfig = {
enabled: false,
instanceUrl: ""
};
const defaultS3Config = {
enabled: false,
forcePathStyle: false
};
const defaultFederationConfig = {
enabled: false,
isPrivate: false,
showEngagement: true,
goLiveMessage: "I've gone live!",
username: "streamer",
blockedDomains: []
};
const defaultHideViewerCount = false;
const defaultSocialHandles = [{
"icon": "/img/platformlogos/github.svg",
"platform": "github",
"url": "https://github.com/owncast/owncast"
}];
const defaultSocialHandlesAdmin = [{
"platform": "github",
"url": "https://github.com/owncast/owncast"
}];
const defaultForbiddenUsernames = [
"owncast",
"operator",
"admin",
"system",
];
const defaultPageContent = `<h1>Welcome to Owncast!</h1>
<ul>
<li>
<p>This is a live stream powered by <a href="https://owncast.online">Owncast</a>, a free and open source live streaming server.</p>
</li>
<li>
<p>To discover more examples of streams, visit <a href="https://directory.owncast.online">Owncast's directory</a>.</p>
</li>
<li>
<p>If you're the owner of this server you should visit the admin and customize the content on this page.</p>
</li>
</ul>
<hr/>
<video id="video" controls preload="metadata" width="40%" poster="https://videos.owncast.online/t/xaJ3xNn9Y6pWTdB25m9ai3">
<source src="https://videos.owncast.online/v/xaJ3xNn9Y6pWTdB25m9ai3.mp4?quality=" type="video/mp4" />
</video>`;
// new configuration for testing
const newServerName = randomString();
const newStreamTitle = randomString();
const newServerSummary = randomString();
const newOfflineMessage = randomString();
const newPageContent = `<p>${randomString()}</p>`;
const newTags = [randomString(), randomString(), randomString()];
const newStreamKeys = [
{ key: randomString(), comment: 'test key 1' },
{ key: randomString(), comment: 'test key 2' },
{ key: randomString(), comment: 'test key 3' },
];
const newAdminPassword = randomString();
const latencyLevel = Random.range(0, 4);
const appearanceValues = {
variable1: randomString(),
variable2: randomString(),
@ -27,52 +95,113 @@ const streamOutputVariants = {
scaledHeight: randomNumber() * 100,
scaledWidth: randomNumber() * 100,
};
const socialHandles = [
const newSocialHandles = [
{
url: 'http://facebook.org/' + randomString(),
platform: randomString(),
},
];
const s3Config = {
enabled: true,
endpoint: 'http://' + randomString(),
const newS3Config = {
enabled: !defaultS3Config.enabled,
endpoint: 'http://' + randomString() + ".tld",
accessKey: randomString(),
secret: randomString(),
bucket: randomString(),
region: randomString(),
forcePathStyle: true,
forcePathStyle: !defaultS3Config.forcePathStyle,
};
const forbiddenUsernames = [randomString(), randomString(), randomString()];
const newForbiddenUsernames = [randomString(), randomString(), randomString()];
const newYPConfig = {
enabled: !defaultYPConfig.enabled,
instanceUrl: 'http://' + randomString() + ".tld"
};
const newFederationConfig = {
enabled: !defaultFederationConfig.enabled,
isPrivate: !defaultFederationConfig.isPrivate,
username: randomString(),
goLiveMessage: randomString(),
showEngagement: !defaultFederationConfig.showEngagement,
blockedDomains: [randomString() + ".tld", randomString() + ".tld"],
};
const newHideViewerCount = !defaultHideViewerCount;
test('verify default config values', async (done) => {
const res = await request.get('/api/config');
expect(res.body.name).toBe(defaultServerName);
expect(res.body.streamTitle).toBe(defaultStreamTitle);
expect(res.body.summary).toBe(`${defaultServerSummary}`);
expect(res.body.extraPageContent).toBe(defaultPageContent);
expect(res.body.offlineMessage).toBe(defaultOfflineMessage);
expect(res.body.logo).toBe(defaultLogo);
expect(res.body.socialHandles).toStrictEqual(defaultSocialHandles);
done();
});
test('verify default admin configuration', async (done) => {
const res = await getAdminConfig();
expect(res.body.instanceDetails.name).toBe(defaultServerName);
expect(res.body.instanceDetails.summary).toBe(defaultServerSummary);
expect(res.body.instanceDetails.offlineMessage).toBe(defaultOfflineMessage);
expect(res.body.instanceDetails.tags).toStrictEqual(defaultTags);
expect(res.body.instanceDetails.socialHandles).toStrictEqual(
defaultSocialHandlesAdmin
);
expect(res.body.forbiddenUsernames).toStrictEqual(defaultForbiddenUsernames);
expect(res.body.streamKeys).toStrictEqual(defaultStreamKeys);
expect(res.body.yp.enabled).toBe(defaultYPConfig.enabled);
expect(res.body.yp.instanceUrl).toBe(defaultYPConfig.instanceUrl);
expect(res.body.adminPassword).toBe(defaultAdminPassword);
expect(res.body.s3.enabled).toBe(defaultS3Config.enabled);
expect(res.body.s3.forcePathStyle).toBe(defaultS3Config.forcePathStyle);
expect(res.body.hideViewerCount).toBe(defaultHideViewerCount);
expect(res.body.federation.enabled).toBe(defaultFederationConfig.enabled);
expect(res.body.federation.username).toBe(defaultFederationConfig.username);
expect(res.body.federation.isPrivate).toBe(defaultFederationConfig.isPrivate);
expect(res.body.federation.showEngagement).toBe(defaultFederationConfig.showEngagement);
expect(res.body.federation.goLiveMessage).toBe(defaultFederationConfig.goLiveMessage);
expect(res.body.federation.blockedDomains).toStrictEqual(defaultFederationConfig.blockedDomains);
done();
});
test('set server name', async (done) => {
const res = await sendConfigChangeRequest('name', serverName);
const res = await sendConfigChangeRequest('name', newServerName);
done();
});
test('set stream title', async (done) => {
const res = await sendConfigChangeRequest('streamtitle', streamTitle);
const res = await sendConfigChangeRequest('streamtitle', newStreamTitle);
done();
});
test('set server summary', async (done) => {
const res = await sendConfigChangeRequest('serversummary', serverSummary);
const res = await sendConfigChangeRequest('serversummary', newServerSummary);
done();
});
test('set extra page content', async (done) => {
const res = await sendConfigChangeRequest('pagecontent', pageContent);
const res = await sendConfigChangeRequest('pagecontent', newPageContent);
done();
});
test('set tags', async (done) => {
const res = await sendConfigChangeRequest('tags', tags);
const res = await sendConfigChangeRequest('tags', newTags);
done();
});
test('set stream keys', async (done) => {
const res = await sendConfigChangeRequest('streamkeys', streamKeys);
const res = await sendConfigChangeRequest('streamkeys', newStreamKeys);
done();
});
@ -92,30 +221,61 @@ test('set video stream output variants', async (done) => {
});
test('set social handles', async (done) => {
const res = await sendConfigChangeRequest('socialhandles', socialHandles);
const res = await sendConfigChangeRequest('socialhandles', newSocialHandles);
done();
});
test('set s3 configuration', async (done) => {
const res = await sendConfigChangeRequest('s3', s3Config);
const res = await sendConfigChangeRequest('s3', newS3Config);
done();
});
test('set forbidden usernames', async (done) => {
const res = await sendConfigChangeRequest(
'chat/forbiddenusernames',
forbiddenUsernames
newForbiddenUsernames
);
done();
});
test('set hide viewer count', async (done) => {
const res = await sendConfigChangeRequest('hideviewercount', true);
test('set server url', async (done) => {
const res = await sendConfigChangeRequest('serverurl', newYPConfig.instanceUrl);
done();
});
test('set federation username', async (done) => {
const res = await sendConfigChangeRequest('federation/username', newFederationConfig.username);
done();
});
test('set federation goLiveMessage', async (done) => {
const res = await sendConfigChangeRequest('federation/livemessage', newFederationConfig.goLiveMessage);
done();
});
test('toggle private federation mode', async (done) => {
const res = await sendConfigChangeRequest('federation/private', newFederationConfig.isPrivate);
done();
});
test('toggle federation engagement', async (done) => {
const res = await sendConfigChangeRequest('federation/showengagement', newFederationConfig.showEngagement);
done();
});
test('set federation blocked domains', async (done) => {
const res = await sendConfigChangeRequest('federation/blockdomains', newFederationConfig.blockedDomains);
done();
});
test('set offline message', async (done) => {
const res = await sendConfigChangeRequest('offlinemessage', offlineMessage);
const res = await sendConfigChangeRequest('offlinemessage', newOfflineMessage);
done();
});
test('set hide viewer count', async (done) => {
const res = await sendConfigChangeRequest('hideviewercount', newHideViewerCount);
done();
});
@ -124,25 +284,49 @@ test('set custom style values', async (done) => {
done();
});
test('enable directory', async (done) => {
const res = await sendConfigChangeRequest('directoryenabled', true);
done();
});
test('enable federation', async (done) => {
const res = await sendConfigChangeRequest('federation/enable', newFederationConfig.enabled);
done();
});
test('change admin password', async (done) => {
const res = await sendConfigChangeRequest('adminpass', newAdminPassword);
done();
});
test('verify admin password change', async (done) => {
const res = await getAdminConfig(adminPassword = newAdminPassword);
expect(res.body.adminPassword).toBe(newAdminPassword);
done();
});
test('reset admin password', async (done) => {
const res = await sendConfigChangeRequest('adminpass', defaultAdminPassword, adminPassword = newAdminPassword);
done();
});
test('verify updated config values', async (done) => {
const res = await request.get('/api/config');
expect(res.body.name).toBe(serverName);
expect(res.body.streamTitle).toBe(streamTitle);
expect(res.body.summary).toBe(`${serverSummary}`);
expect(res.body.extraPageContent).toBe(pageContent);
expect(res.body.offlineMessage).toBe(offlineMessage);
expect(res.body.name).toBe(newServerName);
expect(res.body.streamTitle).toBe(newStreamTitle);
expect(res.body.summary).toBe(`${newServerSummary}`);
expect(res.body.extraPageContent).toBe(newPageContent);
expect(res.body.offlineMessage).toBe(newOfflineMessage);
expect(res.body.logo).toBe('/logo');
expect(res.body.socialHandles).toStrictEqual(socialHandles);
expect(res.body.socialHandles).toStrictEqual(newSocialHandles);
done();
});
// Test that the raw video details being broadcasted are coming through
test('admin stream details are correct', (done) => {
request
.get('/api/admin/status')
.auth('admin', 'abc123')
.expect(200)
.then((res) => {
test('verify admin stream details', async (done) => {
const res = await getAdminStatus();
expect(res.body.broadcaster.streamDetails.width).toBe(320);
expect(res.body.broadcaster.streamDetails.height).toBe(180);
expect(res.body.broadcaster.streamDetails.framerate).toBe(24);
@ -151,24 +335,20 @@ test('admin stream details are correct', (done) => {
expect(res.body.broadcaster.streamDetails.audioCodec).toBe('AAC');
expect(res.body.online).toBe(true);
done();
});
});
test('admin configuration is correct', (done) => {
request
.get('/api/admin/serverconfig')
.auth('admin', 'abc123')
.expect(200)
.then((res) => {
expect(res.body.instanceDetails.name).toBe(serverName);
expect(res.body.instanceDetails.summary).toBe(serverSummary);
expect(res.body.instanceDetails.offlineMessage).toBe(offlineMessage);
expect(res.body.instanceDetails.tags).toStrictEqual(tags);
test('verify updated admin configuration', async (done) => {
const res = await getAdminConfig();
expect(res.body.instanceDetails.name).toBe(newServerName);
expect(res.body.instanceDetails.summary).toBe(newServerSummary);
expect(res.body.instanceDetails.offlineMessage).toBe(newOfflineMessage);
expect(res.body.instanceDetails.tags).toStrictEqual(newTags);
expect(res.body.instanceDetails.socialHandles).toStrictEqual(
socialHandles
newSocialHandles
);
expect(res.body.forbiddenUsernames).toStrictEqual(forbiddenUsernames);
expect(res.body.streamKeys).toStrictEqual(streamKeys);
expect(res.body.forbiddenUsernames).toStrictEqual(newForbiddenUsernames);
expect(res.body.streamKeys).toStrictEqual(newStreamKeys);
expect(res.body.videoSettings.latencyLevel).toBe(latencyLevel);
expect(res.body.videoSettings.videoQualityVariants[0].framerate).toBe(
@ -178,34 +358,43 @@ test('admin configuration is correct', (done) => {
streamOutputVariants.cpuUsageLevel
);
expect(res.body.yp.enabled).toBe(false);
expect(res.body.adminPassword).toBe('abc123');
expect(res.body.yp.enabled).toBe(newYPConfig.enabled);
expect(res.body.yp.instanceUrl).toBe(newYPConfig.instanceUrl);
expect(res.body.s3.enabled).toBe(s3Config.enabled);
expect(res.body.s3.endpoint).toBe(s3Config.endpoint);
expect(res.body.s3.accessKey).toBe(s3Config.accessKey);
expect(res.body.s3.secret).toBe(s3Config.secret);
expect(res.body.s3.bucket).toBe(s3Config.bucket);
expect(res.body.s3.region).toBe(s3Config.region);
expect(res.body.s3.forcePathStyle).toBe(true);
expect(res.body.hideViewerCount).toBe(true);
expect(res.body.adminPassword).toBe(defaultAdminPassword);
expect(res.body.s3.enabled).toBe(newS3Config.enabled);
expect(res.body.s3.endpoint).toBe(newS3Config.endpoint);
expect(res.body.s3.accessKey).toBe(newS3Config.accessKey);
expect(res.body.s3.secret).toBe(newS3Config.secret);
expect(res.body.s3.bucket).toBe(newS3Config.bucket);
expect(res.body.s3.region).toBe(newS3Config.region);
expect(res.body.s3.forcePathStyle).toBe(newS3Config.forcePathStyle);
expect(res.body.hideViewerCount).toBe(newHideViewerCount);
expect(res.body.federation.enabled).toBe(newFederationConfig.enabled);
expect(res.body.federation.isPrivate).toBe(newFederationConfig.isPrivate);
expect(res.body.federation.username).toBe(newFederationConfig.username);
expect(res.body.federation.goLiveMessage).toBe(newFederationConfig.goLiveMessage);
expect(res.body.federation.showEngagement).toBe(newFederationConfig.showEngagement);
expect(res.body.federation.blockedDomains).toStrictEqual(newFederationConfig.blockedDomains);
done();
});
});
test('frontend configuration is correct', (done) => {
test('verify updated frontend configuration', (done) => {
request
.get('/api/config')
.expect(200)
.then((res) => {
expect(res.body.name).toBe(serverName);
expect(res.body.name).toBe(newServerName);
expect(res.body.logo).toBe('/logo');
expect(res.body.socialHandles).toStrictEqual(socialHandles);
expect(res.body.socialHandles).toStrictEqual(newSocialHandles);
done();
});
});
test('frontend status is correct', (done) => {
test('verify frontend status', (done) => {
request
.get('/api/status')
.expect(200)
@ -215,35 +404,11 @@ test('frontend status is correct', (done) => {
});
});
async function sendConfigChangeRequest(endpoint, value) {
const url = '/api/admin/config/' + endpoint;
const res = await request
.post(url)
.auth('admin', 'abc123')
.send({ value: value })
.expect(200);
expect(res.body.success).toBe(true);
return res;
}
async function sendConfigChangePayload(endpoint, payload) {
const url = '/api/admin/config/' + endpoint;
const res = await request
.post(url)
.auth('admin', 'abc123')
.send(payload)
.expect(200);
expect(res.body.success).toBe(true);
return res;
}
function randomString(length = 20) {
return Math.random().toString(16).substr(2, length);
return Random.value().toString(16).substr(2, length);
}
function randomNumber() {
return Math.floor(Math.random() * 5);
return Random.range(0, 5);
}

View file

@ -0,0 +1,134 @@
var request = require('supertest')
const jsonfile = require('jsonfile')
const Ajv = require("ajv-draft-04")
const sendConfigChangeRequest = require('./lib/admin').sendConfigChangeRequest;
request = request('http://127.0.0.1:8080');
var ajv = new Ajv();
var nodeInfoSchema = jsonfile.readFileSync('schema/nodeinfo_2.0.json');
test('disable federation', async (done) => {
const res = await sendConfigChangeRequest('federation/enable', false);
done();
});
test('verify responses of /.well-known/webfinger when federation is disabled', async (done) => {
const res = request
.get('/.well-known/webfinger')
.expect(405);
done();
});
test('verify responses of /.well-known/host-meta when federation is disabled', async (done) => {
const res = request
.get('/.well-known/host-meta')
.expect(405);
done();
});
test('verify responses of /.well-known/nodeinfo when federation is disabled', async (done) => {
const res = request
.get('/.well-known/nodeinfo')
.expect(405);
done();
});
test('verify responses of /.well-known/x-nodeinfo2 when federation is disabled', async (done) => {
const res = request
.get('/.well-known/x-nodeinfo2')
.expect(405);
done();
});
test('verify responses of /nodeinfo/2.0 when federation is disabled', async (done) => {
const res = request
.get('/nodeinfo/2.0')
.expect(405);
done();
});
test('verify responses of /api/v1/instance when federation is disabled', async (done) => {
const res = request
.get('/api/v1/instance')
.expect(405);
done();
});
test('verify responses of /federation/user/ when federation is disabled', async (done) => {
const res = request
.get('/federation/user/')
.expect(405);
done();
});
test('verify responses of /federation/ when federation is disabled', async (done) => {
const res = request
.get('/federation/')
.expect(405);
done();
});
test('enable federation', async (done) => {
const res = await sendConfigChangeRequest('federation/enable', true);
done();
});
test('verify responses of /.well-known/webfinger when federation is enabled', async (done) => {
const res = request
.get('/.well-known/webfinger')
.expect(200);
done();
});
test('verify responses of /.well-known/host-meta when federation is enabled', async (done) => {
const res = request
.get('/.well-known/host-meta')
.expect(200);
done();
});
test('verify responses of /.well-known/nodeinfo when federation is enabled', async (done) => {
const res = request
.get('/.well-known/nodeinfo')
.expect(200);
done();
});
test('verify responses of /.well-known/x-nodeinfo2 when federation is enabled', async (done) => {
const res = request
.get('/.well-known/x-nodeinfo2')
.expect(200);
done();
});
test('verify responses of /nodeinfo/2.0 when federation is enabled', async (done) => {
const res = request
.get('/nodeinfo/2.0')
.expect(200)
.then((res) => {
expect(ajv.validate(nodeInfoSchema, res.body)).toBe(true);
done();
});
});
test('verify responses of /api/v1/instance when federation is enabled', async (done) => {
const res = request
.get('/api/v1/instance')
.expect(200);
done();
});
test('verify responses of /federation/user/ when federation is enabled', async (done) => {
const res = request
.get('/federation/user/')
.expect(200);
done();
});
test('verify responses of /federation/ when federation is enabled', async (done) => {
const res = request
.get('/federation/')
.expect(200);
done();
});

View file

@ -0,0 +1,52 @@
var request = require('supertest');
request = request('http://127.0.0.1:8080');
const defaultAdminPassword = 'abc123';
async function getAdminConfig(adminPassword = defaultAdminPassword) {
const res = request
.get('/api/admin/serverconfig')
.auth('admin', adminPassword)
.expect(200);
return res;
}
async function getAdminStatus(adminPassword = defaultAdminPassword) {
const res = request
.get('/api/admin/status')
.auth('admin', adminPassword)
.expect(200);
return res;
}
async function sendConfigChangeRequest(endpoint, value, adminPassword = defaultAdminPassword) {
const url = '/api/admin/config/' + endpoint;
const res = await request
.post(url)
.auth('admin', adminPassword)
.send({ value: value })
.expect(200);
expect(res.body.success).toBe(true);
return res;
}
async function sendConfigChangePayload(endpoint, payload, adminPassword = defaultAdminPassword) {
const url = '/api/admin/config/' + endpoint;
const res = await request
.post(url)
.auth('admin', adminPassword)
.send(payload)
.expect(200);
expect(res.body.success).toBe(true);
return res;
}
module.exports.getAdminConfig = getAdminConfig;
module.exports.getAdminStatus = getAdminStatus;
module.exports.sendConfigChangeRequest = sendConfigChangeRequest;
module.exports.sendConfigChangePayload = sendConfigChangePayload;

View file

@ -10,7 +10,11 @@
"license": "ISC",
"dependencies": {
"supertest": "^6.0.1",
"websocket": "^1.0.32"
"websocket": "^1.0.32",
"ajv": "^8.11.0",
"ajv-draft-04" : "^1.0.0",
"jsonfile": "^6.1.0",
"crypto-random": "^2.0.1"
},
"devDependencies": {
"jest": "^26.6.3"

View file

@ -3,7 +3,7 @@
TEMP_DB=$(mktemp)
# Install the node test framework
npm install #--silent >/dev/null
npm install --quiet --no-progress
# Download a specific version of ffmpeg
if [ ! -d "ffmpeg" ]; then

View file

@ -0,0 +1,180 @@
{
"$schema": "http://json-schema.org/draft-04/schema#",
"id": "http://nodeinfo.diaspora.software/ns/schema/2.0#",
"description": "NodeInfo schema version 2.0.",
"type": "object",
"additionalProperties": false,
"required": [
"version",
"software",
"protocols",
"services",
"openRegistrations",
"usage",
"metadata"
],
"properties": {
"version": {
"description": "The schema version, must be 2.0.",
"enum": [
"2.0"
]
},
"software": {
"description": "Metadata about server software in use.",
"type": "object",
"additionalProperties": false,
"required": [
"name",
"version"
],
"properties": {
"name": {
"description": "The canonical name of this server software.",
"type": "string",
"pattern": "^[a-z0-9-]+$"
},
"version": {
"description": "The version of this server software.",
"type": "string"
}
}
},
"protocols": {
"description": "The protocols supported on this server.",
"type": "array",
"minItems": 1,
"items": {
"enum": [
"activitypub",
"buddycloud",
"dfrn",
"diaspora",
"libertree",
"ostatus",
"pumpio",
"tent",
"xmpp",
"zot"
]
}
},
"services": {
"description": "The third party sites this server can connect to via their application API.",
"type": "object",
"additionalProperties": false,
"required": [
"inbound",
"outbound"
],
"properties": {
"inbound": {
"description": "The third party sites this server can retrieve messages from for combined display with regular traffic.",
"type": "array",
"minItems": 0,
"items": {
"enum": [
"atom1.0",
"gnusocial",
"imap",
"pnut",
"pop3",
"pumpio",
"rss2.0",
"twitter"
]
}
},
"outbound": {
"description": "The third party sites this server can publish messages to on the behalf of a user.",
"type": "array",
"minItems": 0,
"items": {
"enum": [
"atom1.0",
"blogger",
"buddycloud",
"diaspora",
"dreamwidth",
"drupal",
"facebook",
"friendica",
"gnusocial",
"google",
"insanejournal",
"libertree",
"linkedin",
"livejournal",
"mediagoblin",
"myspace",
"pinterest",
"pnut",
"posterous",
"pumpio",
"redmatrix",
"rss2.0",
"smtp",
"tent",
"tumblr",
"twitter",
"wordpress",
"xmpp"
]
}
}
}
},
"openRegistrations": {
"description": "Whether this server allows open self-registration.",
"type": "boolean"
},
"usage": {
"description": "Usage statistics for this server.",
"type": "object",
"additionalProperties": false,
"required": [
"users"
],
"properties": {
"users": {
"description": "statistics about the users of this server.",
"type": "object",
"additionalProperties": false,
"properties": {
"total": {
"description": "The total amount of on this server registered users.",
"type": "integer",
"minimum": 0
},
"activeHalfyear": {
"description": "The amount of users that signed in at least once in the last 180 days.",
"type": "integer",
"minimum": 0
},
"activeMonth": {
"description": "The amount of users that signed in at least once in the last 30 days.",
"type": "integer",
"minimum": 0
}
}
},
"localPosts": {
"description": "The amount of posts that were made by users that are registered on this server.",
"type": "integer",
"minimum": 0
},
"localComments": {
"description": "The amount of comments that were made by users that are registered on this server.",
"type": "integer",
"minimum": 0
}
}
},
"metadata": {
"description": "Free form key value pairs for software specific values. Clients should not rely on any specific key present.",
"type": "object",
"minProperties": 0,
"additionalProperties": true
}
}
}

View file

@ -17,7 +17,7 @@ echo "Bundling web code into server..."
# Install the web test framework
echo "Installing test dependencies..."
pushd test/automated/browser
npm install --silent >/dev/null
npm install --quiet --no-progress
popd