forked from FoundKeyGang/FoundKey
nanka iroiro
This commit is contained in:
parent
f93eb00207
commit
4e5545af38
17 changed files with 472 additions and 205 deletions
|
@ -243,6 +243,10 @@ desktop:
|
||||||
title: "Notifications"
|
title: "Notifications"
|
||||||
settings: "Notification settings"
|
settings: "Notification settings"
|
||||||
|
|
||||||
|
mk-server-home-widget:
|
||||||
|
title: "Server info"
|
||||||
|
toggle: "Toggle views"
|
||||||
|
|
||||||
mk-activity-home-widget:
|
mk-activity-home-widget:
|
||||||
title: "Activity"
|
title: "Activity"
|
||||||
toggle: "Toggle views"
|
toggle: "Toggle views"
|
||||||
|
|
|
@ -243,6 +243,10 @@ desktop:
|
||||||
title: "通知"
|
title: "通知"
|
||||||
settings: "通知の設定"
|
settings: "通知の設定"
|
||||||
|
|
||||||
|
mk-server-home-widget:
|
||||||
|
title: "サーバー情報"
|
||||||
|
toggle: "表示を切り替え"
|
||||||
|
|
||||||
mk-activity-home-widget:
|
mk-activity-home-widget:
|
||||||
title: "アクティビティ"
|
title: "アクティビティ"
|
||||||
toggle: "表示を切り替え"
|
toggle: "表示を切り替え"
|
||||||
|
|
|
@ -106,6 +106,7 @@
|
||||||
"debug": "2.6.8",
|
"debug": "2.6.8",
|
||||||
"deep-equal": "1.0.1",
|
"deep-equal": "1.0.1",
|
||||||
"deepcopy": "0.6.3",
|
"deepcopy": "0.6.3",
|
||||||
|
"diskusage": "^0.2.2",
|
||||||
"download": "6.2.2",
|
"download": "6.2.2",
|
||||||
"elasticsearch": "13.0.1",
|
"elasticsearch": "13.0.1",
|
||||||
"escape-regexp": "0.0.1",
|
"escape-regexp": "0.0.1",
|
||||||
|
@ -123,6 +124,7 @@
|
||||||
"ms": "2.0.0",
|
"ms": "2.0.0",
|
||||||
"multer": "1.3.0",
|
"multer": "1.3.0",
|
||||||
"nprogress": "0.2.0",
|
"nprogress": "0.2.0",
|
||||||
|
"os-utils": "0.0.14",
|
||||||
"page": "1.7.1",
|
"page": "1.7.1",
|
||||||
"pictograph": "2.0.4",
|
"pictograph": "2.0.4",
|
||||||
"prominence": "0.2.0",
|
"prominence": "0.2.0",
|
||||||
|
@ -145,6 +147,7 @@
|
||||||
"typescript": "2.3.4",
|
"typescript": "2.3.4",
|
||||||
"uuid": "3.0.1",
|
"uuid": "3.0.1",
|
||||||
"vhost": "3.0.2",
|
"vhost": "3.0.2",
|
||||||
"websocket": "1.0.24"
|
"websocket": "1.0.24",
|
||||||
|
"xev": "^2.0.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
20
src/api/stream/server.ts
Normal file
20
src/api/stream/server.ts
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
import * as websocket from 'websocket';
|
||||||
|
import Xev from 'xev';
|
||||||
|
|
||||||
|
const ev = new Xev();
|
||||||
|
|
||||||
|
export default function homeStream(request: websocket.request, connection: websocket.connection): void {
|
||||||
|
const onStats = stats => {
|
||||||
|
connection.send(JSON.stringify({
|
||||||
|
type: 'stats',
|
||||||
|
body: stats
|
||||||
|
}));
|
||||||
|
};
|
||||||
|
|
||||||
|
ev.addListener('stats', onStats);
|
||||||
|
|
||||||
|
connection.on('close', () => {
|
||||||
|
console.log('yooo');
|
||||||
|
ev.removeListener('stats', onStats);
|
||||||
|
});
|
||||||
|
}
|
|
@ -8,6 +8,7 @@ import isNativeToken from './common/is-native-token';
|
||||||
|
|
||||||
import homeStream from './stream/home';
|
import homeStream from './stream/home';
|
||||||
import messagingStream from './stream/messaging';
|
import messagingStream from './stream/messaging';
|
||||||
|
import serverStream from './stream/server';
|
||||||
|
|
||||||
module.exports = (server: http.Server) => {
|
module.exports = (server: http.Server) => {
|
||||||
/**
|
/**
|
||||||
|
@ -20,6 +21,11 @@ module.exports = (server: http.Server) => {
|
||||||
ws.on('request', async (request) => {
|
ws.on('request', async (request) => {
|
||||||
const connection = request.accept();
|
const connection = request.accept();
|
||||||
|
|
||||||
|
if (request.resourceURL.pathname === '/server') {
|
||||||
|
serverStream(request, connection);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const user = await authenticate(connection, request.resourceURL.query.i);
|
const user = await authenticate(connection, request.resourceURL.query.i);
|
||||||
|
|
||||||
if (user == null) {
|
if (user == null) {
|
||||||
|
|
|
@ -12,17 +12,20 @@ import * as chalk from 'chalk';
|
||||||
// import portUsed = require('tcp-port-used');
|
// import portUsed = require('tcp-port-used');
|
||||||
import isRoot = require('is-root');
|
import isRoot = require('is-root');
|
||||||
import { master } from 'accesses';
|
import { master } from 'accesses';
|
||||||
|
import Xev from 'xev';
|
||||||
|
|
||||||
import Logger from './utils/logger';
|
import Logger from './utils/logger';
|
||||||
import ProgressBar from './utils/cli/progressbar';
|
import ProgressBar from './utils/cli/progressbar';
|
||||||
import EnvironmentInfo from './utils/environmentInfo';
|
import EnvironmentInfo from './utils/environmentInfo';
|
||||||
import MachineInfo from './utils/machineInfo';
|
import MachineInfo from './utils/machineInfo';
|
||||||
import DependencyInfo from './utils/dependencyInfo';
|
import DependencyInfo from './utils/dependencyInfo';
|
||||||
|
import stats from './utils/stats';
|
||||||
|
|
||||||
import { Config, path as configPath } from './config';
|
import { Config, path as configPath } from './config';
|
||||||
import loadConfig from './config';
|
import loadConfig from './config';
|
||||||
|
|
||||||
const clusterLog = debug('misskey:cluster');
|
const clusterLog = debug('misskey:cluster');
|
||||||
|
const ev = new Xev();
|
||||||
|
|
||||||
process.title = 'Misskey';
|
process.title = 'Misskey';
|
||||||
|
|
||||||
|
@ -35,6 +38,9 @@ main();
|
||||||
function main() {
|
function main() {
|
||||||
if (cluster.isMaster) {
|
if (cluster.isMaster) {
|
||||||
masterMain();
|
masterMain();
|
||||||
|
|
||||||
|
ev.mount();
|
||||||
|
stats();
|
||||||
} else {
|
} else {
|
||||||
workerMain();
|
workerMain();
|
||||||
}
|
}
|
||||||
|
|
25
src/utils/stats.ts
Normal file
25
src/utils/stats.ts
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
import * as os from 'os';
|
||||||
|
const osUtils = require('os-utils');
|
||||||
|
import * as diskusage from 'diskusage';
|
||||||
|
import Xev from 'xev';
|
||||||
|
|
||||||
|
const ev = new Xev();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Report stats regularly
|
||||||
|
*/
|
||||||
|
export default function() {
|
||||||
|
setInterval(() => {
|
||||||
|
osUtils.cpuUsage(cpuUsage => {
|
||||||
|
const disk = diskusage.checkSync(os.platform() == 'win32' ? 'c:' : '/');
|
||||||
|
ev.emit('stats', {
|
||||||
|
cpu_usage: cpuUsage,
|
||||||
|
mem: {
|
||||||
|
total: os.totalmem(),
|
||||||
|
free: os.freemem()
|
||||||
|
},
|
||||||
|
disk
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}, 1000);
|
||||||
|
}
|
18
src/web/app/common/scripts/home-stream.js
Normal file
18
src/web/app/common/scripts/home-stream.js
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
import Stream from './stream';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Home stream connection
|
||||||
|
*/
|
||||||
|
class Connection extends Stream {
|
||||||
|
constructor(me) {
|
||||||
|
super('', {
|
||||||
|
i: me.token
|
||||||
|
});
|
||||||
|
|
||||||
|
this.on('i_updated', me.update);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Connection;
|
|
@ -1,42 +1,22 @@
|
||||||
const ReconnectingWebSocket = require('reconnecting-websocket');
|
'use strict';
|
||||||
import * as riot from 'riot';
|
|
||||||
import CONFIG from './config';
|
|
||||||
|
|
||||||
class Connection {
|
import Stream from './stream';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Messaging stream connection
|
||||||
|
*/
|
||||||
|
class Connection extends Stream {
|
||||||
constructor(me, otherparty) {
|
constructor(me, otherparty) {
|
||||||
// BIND -----------------------------------
|
super('messaging', {
|
||||||
this.onOpen = this.onOpen.bind(this);
|
i: me.token,
|
||||||
this.onMessage = this.onMessage.bind(this);
|
otherparty
|
||||||
this.close = this.close.bind(this);
|
});
|
||||||
// ----------------------------------------
|
|
||||||
|
|
||||||
this.event = riot.observable();
|
this.on('_connected_', () => {
|
||||||
this.me = me;
|
this.send({
|
||||||
|
i: me.token
|
||||||
const host = CONFIG.apiUrl.replace('http', 'ws');
|
});
|
||||||
this.socket = new ReconnectingWebSocket(`${host}/messaging?i=${me.token}&otherparty=${otherparty}`);
|
});
|
||||||
this.socket.addEventListener('open', this.onOpen);
|
|
||||||
this.socket.addEventListener('message', this.onMessage);
|
|
||||||
}
|
|
||||||
|
|
||||||
onOpen() {
|
|
||||||
this.socket.send(JSON.stringify({
|
|
||||||
i: this.me.token
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
onMessage(message) {
|
|
||||||
try {
|
|
||||||
const msg = JSON.parse(message.data);
|
|
||||||
if (msg.type) this.event.trigger(msg.type, msg.body);
|
|
||||||
} catch(e) {
|
|
||||||
// noop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
close() {
|
|
||||||
this.socket.removeEventListener('open', this.onOpen);
|
|
||||||
this.socket.removeEventListener('message', this.onMessage);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
14
src/web/app/common/scripts/server-stream.js
Normal file
14
src/web/app/common/scripts/server-stream.js
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
import Stream from './stream';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Server stream connection
|
||||||
|
*/
|
||||||
|
class Connection extends Stream {
|
||||||
|
constructor() {
|
||||||
|
super('server');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Connection;
|
|
@ -5,10 +5,10 @@ import * as riot from 'riot';
|
||||||
import CONFIG from './config';
|
import CONFIG from './config';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Home stream connection
|
* Misskey stream connection
|
||||||
*/
|
*/
|
||||||
class Connection {
|
class Connection {
|
||||||
constructor(me) {
|
constructor(endpoint, params) {
|
||||||
// BIND -----------------------------------
|
// BIND -----------------------------------
|
||||||
this.onOpen = this.onOpen.bind(this);
|
this.onOpen = this.onOpen.bind(this);
|
||||||
this.onClose = this.onClose.bind(this);
|
this.onClose = this.onClose.bind(this);
|
||||||
|
@ -20,16 +20,19 @@ class Connection {
|
||||||
riot.observable(this);
|
riot.observable(this);
|
||||||
|
|
||||||
this.state = 'initializing';
|
this.state = 'initializing';
|
||||||
this.me = me;
|
|
||||||
this.buffer = [];
|
this.buffer = [];
|
||||||
|
|
||||||
const host = CONFIG.apiUrl.replace('http', 'ws');
|
const host = CONFIG.apiUrl.replace('http', 'ws');
|
||||||
this.socket = new ReconnectingWebSocket(`${host}?i=${me.token}`);
|
const query = params
|
||||||
|
? Object.keys(params)
|
||||||
|
.map(k => encodeURIComponent(k) + '=' + encodeURIComponent(params[k]))
|
||||||
|
.join('&')
|
||||||
|
: null;
|
||||||
|
|
||||||
|
this.socket = new ReconnectingWebSocket(`${host}/${endpoint}${query ? '?' + query : ''}`);
|
||||||
this.socket.addEventListener('open', this.onOpen);
|
this.socket.addEventListener('open', this.onOpen);
|
||||||
this.socket.addEventListener('close', this.onClose);
|
this.socket.addEventListener('close', this.onClose);
|
||||||
this.socket.addEventListener('message', this.onMessage);
|
this.socket.addEventListener('message', this.onMessage);
|
||||||
|
|
||||||
this.on('i_updated', me.update);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -137,8 +137,8 @@
|
||||||
this.connection = new MessagingStreamConnection(this.I, this.user.id);
|
this.connection = new MessagingStreamConnection(this.I, this.user.id);
|
||||||
|
|
||||||
this.on('mount', () => {
|
this.on('mount', () => {
|
||||||
this.connection.event.on('message', this.onMessage);
|
this.connection.on('message', this.onMessage);
|
||||||
this.connection.event.on('read', this.onRead);
|
this.connection.on('read', this.onRead);
|
||||||
|
|
||||||
document.addEventListener('visibilitychange', this.onVisibilitychange);
|
document.addEventListener('visibilitychange', this.onVisibilitychange);
|
||||||
|
|
||||||
|
@ -153,8 +153,8 @@
|
||||||
});
|
});
|
||||||
|
|
||||||
this.on('unmount', () => {
|
this.on('unmount', () => {
|
||||||
this.connection.event.off('message', this.onMessage);
|
this.connection.off('message', this.onMessage);
|
||||||
this.connection.event.off('read', this.onRead);
|
this.connection.off('read', this.onRead);
|
||||||
this.connection.close();
|
this.connection.close();
|
||||||
|
|
||||||
document.removeEventListener('visibilitychange', this.onVisibilitychange);
|
document.removeEventListener('visibilitychange', this.onVisibilitychange);
|
||||||
|
@ -174,10 +174,10 @@
|
||||||
|
|
||||||
this.messages.push(message);
|
this.messages.push(message);
|
||||||
if (message.user_id != this.I.id && !document.hidden) {
|
if (message.user_id != this.I.id && !document.hidden) {
|
||||||
this.connection.socket.send(JSON.stringify({
|
this.connection.send({
|
||||||
type: 'read',
|
type: 'read',
|
||||||
id: message.id
|
id: message.id
|
||||||
}));
|
});
|
||||||
}
|
}
|
||||||
this.update();
|
this.update();
|
||||||
|
|
||||||
|
@ -239,10 +239,10 @@
|
||||||
if (document.hidden) return;
|
if (document.hidden) return;
|
||||||
this.messages.forEach(message => {
|
this.messages.forEach(message => {
|
||||||
if (message.user_id !== this.I.id && !message.is_read) {
|
if (message.user_id !== this.I.id && !message.is_read) {
|
||||||
this.connection.socket.send(JSON.stringify({
|
this.connection.send({
|
||||||
type: 'read',
|
type: 'read',
|
||||||
id: message.id
|
id: message.id
|
||||||
}));
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
|
@ -100,6 +100,8 @@
|
||||||
</svg>
|
</svg>
|
||||||
<style>
|
<style>
|
||||||
:scope
|
:scope
|
||||||
|
display block
|
||||||
|
|
||||||
> svg
|
> svg
|
||||||
display block
|
display block
|
||||||
padding 10px
|
padding 10px
|
||||||
|
@ -131,7 +133,7 @@
|
||||||
</mk-activity-home-widget-calender>
|
</mk-activity-home-widget-calender>
|
||||||
|
|
||||||
<mk-activity-home-widget-chart>
|
<mk-activity-home-widget-chart>
|
||||||
<svg riot-viewBox="0 0 { viewBoxX } 60" preserveAspectRatio="none" onmousedown={ onMousedown }>
|
<svg riot-viewBox="0 0 { viewBoxX } { viewBoxY }" preserveAspectRatio="none" onmousedown={ onMousedown }>
|
||||||
<polyline
|
<polyline
|
||||||
riot-points={ pointsPost }
|
riot-points={ pointsPost }
|
||||||
fill="none"
|
fill="none"
|
||||||
|
@ -155,6 +157,8 @@
|
||||||
</svg>
|
</svg>
|
||||||
<style>
|
<style>
|
||||||
:scope
|
:scope
|
||||||
|
display block
|
||||||
|
|
||||||
> svg
|
> svg
|
||||||
display block
|
display block
|
||||||
padding 10px
|
padding 10px
|
||||||
|
@ -163,6 +167,7 @@
|
||||||
</style>
|
</style>
|
||||||
<script>
|
<script>
|
||||||
this.viewBoxX = 140;
|
this.viewBoxX = 140;
|
||||||
|
this.viewBoxY = 60;
|
||||||
this.zoom = 1;
|
this.zoom = 1;
|
||||||
this.pos = 0;
|
this.pos = 0;
|
||||||
|
|
||||||
|
@ -176,10 +181,10 @@
|
||||||
|
|
||||||
this.render = () => {
|
this.render = () => {
|
||||||
this.update({
|
this.update({
|
||||||
pointsPost: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.posts / peak)) * 60}`).join(' '),
|
pointsPost: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.posts / peak)) * this.viewBoxY}`).join(' '),
|
||||||
pointsReply: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.replies / peak)) * 60}`).join(' '),
|
pointsReply: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.replies / peak)) * this.viewBoxY}`).join(' '),
|
||||||
pointsRepost: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.reposts / peak)) * 60}`).join(' '),
|
pointsRepost: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.reposts / peak)) * this.viewBoxY}`).join(' '),
|
||||||
pointsTotal: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.total / peak)) * 60}`).join(' ')
|
pointsTotal: this.data.map((d, i) => `${(i * this.zoom) + this.pos},${(1 - (d.total / peak)) * this.viewBoxY}`).join(' ')
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
177
src/web/app/desktop/tags/home-widgets/server.tag
Normal file
177
src/web/app/desktop/tags/home-widgets/server.tag
Normal file
|
@ -0,0 +1,177 @@
|
||||||
|
<mk-server-home-widget>
|
||||||
|
<p class="title"><i class="fa fa-server"></i>%i18n:desktop.tags.mk-server-home-widget.title%</p>
|
||||||
|
<button onclick={ toggle } title="%i18n:desktop.tags.mk-server-home-widget.toggle%"><i class="fa fa-sort"></i></button>
|
||||||
|
<p class="initializing" if={ initializing }><i class="fa fa-spinner fa-pulse fa-fw"></i>%i18n:common.loading%<mk-ellipsis/></p>
|
||||||
|
<mk-server-home-widget-stats if={ !initializing && view == 0 }/>
|
||||||
|
<mk-server-home-widget-info if={ !initializing && view == 1 } meta={ meta }/>
|
||||||
|
<style>
|
||||||
|
:scope
|
||||||
|
display block
|
||||||
|
background #fff
|
||||||
|
|
||||||
|
> .title
|
||||||
|
z-index 1
|
||||||
|
margin 0
|
||||||
|
padding 0 16px
|
||||||
|
line-height 42px
|
||||||
|
font-size 0.9em
|
||||||
|
font-weight bold
|
||||||
|
color #888
|
||||||
|
box-shadow 0 1px rgba(0, 0, 0, 0.07)
|
||||||
|
|
||||||
|
> i
|
||||||
|
margin-right 4px
|
||||||
|
|
||||||
|
> button
|
||||||
|
position absolute
|
||||||
|
z-index 2
|
||||||
|
top 0
|
||||||
|
right 0
|
||||||
|
padding 0
|
||||||
|
width 42px
|
||||||
|
font-size 0.9em
|
||||||
|
line-height 42px
|
||||||
|
color #ccc
|
||||||
|
|
||||||
|
&:hover
|
||||||
|
color #aaa
|
||||||
|
|
||||||
|
&:active
|
||||||
|
color #999
|
||||||
|
|
||||||
|
> .initializing
|
||||||
|
margin 0
|
||||||
|
padding 16px
|
||||||
|
text-align center
|
||||||
|
color #aaa
|
||||||
|
|
||||||
|
> i
|
||||||
|
margin-right 4px
|
||||||
|
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
this.mixin('api');
|
||||||
|
|
||||||
|
this.initializing = true;
|
||||||
|
this.view = 0;
|
||||||
|
|
||||||
|
this.on('mount', () => {
|
||||||
|
this.api('meta').then(meta => {
|
||||||
|
this.update({
|
||||||
|
initializing: false,
|
||||||
|
meta
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
this.toggle = () => {
|
||||||
|
this.view++;
|
||||||
|
if (this.view == 2) this.view = 0;
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
</mk-server-home-widget>
|
||||||
|
|
||||||
|
<mk-server-home-widget-stats>
|
||||||
|
<svg riot-viewBox="0 0 { viewBoxX } { viewBoxY }" preserveAspectRatio="none">
|
||||||
|
<text dx="1" dy="5">CPU</text>
|
||||||
|
<polygon
|
||||||
|
riot-points={ cpuPolygonPoints }
|
||||||
|
riot-fill={ cpuColor }
|
||||||
|
fill-opacity="0.5"/>
|
||||||
|
<polyline
|
||||||
|
riot-points={ cpuPolylinePoints }
|
||||||
|
fill="none"
|
||||||
|
stroke-width="1"
|
||||||
|
riot-stroke={ cpuColor }/>
|
||||||
|
</svg>
|
||||||
|
<svg riot-viewBox="0 0 { viewBoxX } { viewBoxY }" preserveAspectRatio="none">
|
||||||
|
<text dx="1" dy="5">MEM</text>
|
||||||
|
<polygon
|
||||||
|
riot-points={ memPolygonPoints }
|
||||||
|
riot-fill={ memColor }
|
||||||
|
fill-opacity="0.5"/>
|
||||||
|
<polyline
|
||||||
|
riot-points={ memPolylinePoints }
|
||||||
|
fill="none"
|
||||||
|
stroke-width="1"
|
||||||
|
riot-stroke={ memColor }/>
|
||||||
|
</svg>
|
||||||
|
<style>
|
||||||
|
:scope
|
||||||
|
display block
|
||||||
|
|
||||||
|
> svg
|
||||||
|
display block
|
||||||
|
padding 10px
|
||||||
|
width 50%
|
||||||
|
float left
|
||||||
|
|
||||||
|
&:first-child
|
||||||
|
padding-right 5px
|
||||||
|
|
||||||
|
&:last-child
|
||||||
|
padding-left 5px
|
||||||
|
|
||||||
|
> text
|
||||||
|
font-size 5px
|
||||||
|
fill #7b7b7b
|
||||||
|
|
||||||
|
&:after
|
||||||
|
content ""
|
||||||
|
display block
|
||||||
|
clear both
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
import Connection from '../../../common/scripts/server-stream';
|
||||||
|
|
||||||
|
this.viewBoxX = 50;
|
||||||
|
this.viewBoxY = 30;
|
||||||
|
this.stats = [];
|
||||||
|
this.connection = new Connection();
|
||||||
|
|
||||||
|
this.on('mount', () => {
|
||||||
|
this.connection.on('stats', this.onStats);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.on('unmount', () => {
|
||||||
|
this.connection.off('stats', this.onStats);
|
||||||
|
this.connection.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
this.onStats = stats => {
|
||||||
|
this.stats.push(stats);
|
||||||
|
if (this.stats.length > 50) this.stats.shift();
|
||||||
|
|
||||||
|
const cpuPolylinePoints = this.stats.map((s, i) => `${this.viewBoxX - ((this.stats.length - 1) - i)},${(1 - s.cpu_usage) * this.viewBoxY}`).join(' ');
|
||||||
|
const memPolylinePoints = this.stats.map((s, i) => `${this.viewBoxX - ((this.stats.length - 1) - i)},${(s.mem.free / s.mem.total) * this.viewBoxY}`).join(' ');
|
||||||
|
|
||||||
|
const cpuPolygonPoints = `${this.viewBoxX - (this.stats.length - 1)},${ this.viewBoxY } ${ cpuPolylinePoints } ${ this.viewBoxX },${ this.viewBoxY }`;
|
||||||
|
const memPolygonPoints = `${this.viewBoxX - (this.stats.length - 1)},${ this.viewBoxY } ${ memPolylinePoints } ${ this.viewBoxX },${ this.viewBoxY }`;
|
||||||
|
|
||||||
|
const cpuColor = `hsl(${180 - (stats.cpu_usage * 180)}, 80%, 70%)`;
|
||||||
|
const memColor = `hsl(${180 - (stats.mem.free / stats.mem.total * 180)}, 80%, 70%)`;
|
||||||
|
|
||||||
|
this.update({
|
||||||
|
cpuPolylinePoints,
|
||||||
|
memPolylinePoints,
|
||||||
|
cpuPolygonPoints,
|
||||||
|
memPolygonPoints,
|
||||||
|
cpuColor,
|
||||||
|
memColor
|
||||||
|
});
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
</mk-server-home-widget-stats>
|
||||||
|
|
||||||
|
<mk-server-home-widget-info>
|
||||||
|
<p>Maintainer: { meta.maintainer }</p>
|
||||||
|
<style>
|
||||||
|
:scope
|
||||||
|
display block
|
||||||
|
padding 10px
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
this.meta = this.opts.meta;
|
||||||
|
</script>
|
||||||
|
</mk-server-home-widget-info>
|
||||||
|
|
|
@ -70,6 +70,7 @@
|
||||||
'rss-reader',
|
'rss-reader',
|
||||||
'trends',
|
'trends',
|
||||||
'photo-stream',
|
'photo-stream',
|
||||||
|
'server',
|
||||||
'version'
|
'version'
|
||||||
],
|
],
|
||||||
right: [
|
right: [
|
||||||
|
|
|
@ -45,6 +45,7 @@ require('./home-widgets/version.tag');
|
||||||
require('./home-widgets/recommended-polls.tag');
|
require('./home-widgets/recommended-polls.tag');
|
||||||
require('./home-widgets/trends.tag');
|
require('./home-widgets/trends.tag');
|
||||||
require('./home-widgets/activity.tag');
|
require('./home-widgets/activity.tag');
|
||||||
|
require('./home-widgets/server.tag');
|
||||||
require('./timeline.tag');
|
require('./timeline.tag');
|
||||||
require('./messaging/window.tag');
|
require('./messaging/window.tag');
|
||||||
require('./messaging/room-window.tag');
|
require('./messaging/room-window.tag');
|
||||||
|
|
|
@ -8,7 +8,7 @@ import * as riot from 'riot';
|
||||||
import api from './common/scripts/api';
|
import api from './common/scripts/api';
|
||||||
import signout from './common/scripts/signout';
|
import signout from './common/scripts/signout';
|
||||||
import checkForUpdate from './common/scripts/check-for-update';
|
import checkForUpdate from './common/scripts/check-for-update';
|
||||||
import Connection from './common/scripts/stream';
|
import Connection from './common/scripts/home-stream';
|
||||||
import mixin from './common/mixins';
|
import mixin from './common/mixins';
|
||||||
import generateDefaultUserdata from './common/scripts/generate-default-userdata';
|
import generateDefaultUserdata from './common/scripts/generate-default-userdata';
|
||||||
import CONFIG from './common/scripts/config';
|
import CONFIG from './common/scripts/config';
|
||||||
|
@ -95,7 +95,7 @@ export default callback => {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Init stream connection
|
// Init home stream connection
|
||||||
const stream = me ? new Connection(me) : null;
|
const stream = me ? new Connection(me) : null;
|
||||||
|
|
||||||
// ミックスイン初期化
|
// ミックスイン初期化
|
||||||
|
|
Loading…
Reference in a new issue