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:
@@ -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,88 +284,117 @@ 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) => {
|
||||
expect(res.body.broadcaster.streamDetails.width).toBe(320);
|
||||
expect(res.body.broadcaster.streamDetails.height).toBe(180);
|
||||
expect(res.body.broadcaster.streamDetails.framerate).toBe(24);
|
||||
expect(res.body.broadcaster.streamDetails.videoBitrate).toBe(1269);
|
||||
expect(res.body.broadcaster.streamDetails.videoCodec).toBe('H.264');
|
||||
expect(res.body.broadcaster.streamDetails.audioCodec).toBe('AAC');
|
||||
expect(res.body.online).toBe(true);
|
||||
done();
|
||||
});
|
||||
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);
|
||||
expect(res.body.broadcaster.streamDetails.videoBitrate).toBe(1269);
|
||||
expect(res.body.broadcaster.streamDetails.videoCodec).toBe('H.264');
|
||||
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);
|
||||
expect(res.body.instanceDetails.socialHandles).toStrictEqual(
|
||||
socialHandles
|
||||
);
|
||||
expect(res.body.forbiddenUsernames).toStrictEqual(forbiddenUsernames);
|
||||
expect(res.body.streamKeys).toStrictEqual(streamKeys);
|
||||
test('verify updated admin configuration', async (done) => {
|
||||
const res = await getAdminConfig();
|
||||
|
||||
expect(res.body.videoSettings.latencyLevel).toBe(latencyLevel);
|
||||
expect(res.body.videoSettings.videoQualityVariants[0].framerate).toBe(
|
||||
streamOutputVariants.framerate
|
||||
);
|
||||
expect(res.body.videoSettings.videoQualityVariants[0].cpuUsageLevel).toBe(
|
||||
streamOutputVariants.cpuUsageLevel
|
||||
);
|
||||
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(
|
||||
newSocialHandles
|
||||
);
|
||||
expect(res.body.forbiddenUsernames).toStrictEqual(newForbiddenUsernames);
|
||||
expect(res.body.streamKeys).toStrictEqual(newStreamKeys);
|
||||
|
||||
expect(res.body.yp.enabled).toBe(false);
|
||||
expect(res.body.adminPassword).toBe('abc123');
|
||||
expect(res.body.videoSettings.latencyLevel).toBe(latencyLevel);
|
||||
expect(res.body.videoSettings.videoQualityVariants[0].framerate).toBe(
|
||||
streamOutputVariants.framerate
|
||||
);
|
||||
expect(res.body.videoSettings.videoQualityVariants[0].cpuUsageLevel).toBe(
|
||||
streamOutputVariants.cpuUsageLevel
|
||||
);
|
||||
|
||||
expect(res.body.yp.enabled).toBe(newYPConfig.enabled);
|
||||
expect(res.body.yp.instanceUrl).toBe(newYPConfig.instanceUrl);
|
||||
|
||||
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();
|
||||
|
||||
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);
|
||||
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);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user