]>
Commit | Line | Data |
---|---|---|
d7139140 DM |
1 | Ext.define('proxmox-services', { |
2 | extend: 'Ext.data.Model', | |
6607de47 | 3 | fields: ['service', 'name', 'desc', 'state', 'unit-state', 'active-state'], |
01031528 | 4 | idProperty: 'service', |
d7139140 DM |
5 | }); |
6 | ||
7 | Ext.define('Proxmox.node.ServiceView', { | |
8 | extend: 'Ext.grid.GridPanel', | |
9 | ||
10 | alias: ['widget.proxmoxNodeServiceView'], | |
11 | ||
12 | startOnlyServices: {}, | |
13 | ||
2815c428 TL |
14 | restartCommand: "restart", // TODO: default to reload once everywhere supported |
15 | ||
01031528 | 16 | initComponent: function() { |
05a977a2 | 17 | let me = this; |
d7139140 DM |
18 | |
19 | if (!me.nodename) { | |
20 | throw "no node name specified"; | |
21 | } | |
22 | ||
05a977a2 | 23 | let rstore = Ext.create('Proxmox.data.UpdateStore', { |
d7139140 | 24 | interval: 1000, |
d7139140 DM |
25 | model: 'proxmox-services', |
26 | proxy: { | |
ab34f5b2 TL |
27 | type: 'proxmox', |
28 | url: `/api2/json/nodes/${me.nodename}/services`, | |
01031528 | 29 | }, |
d7139140 DM |
30 | }); |
31 | ||
05a977a2 | 32 | let store = Ext.create('Proxmox.data.DiffStore', { |
d7139140 DM |
33 | rstore: rstore, |
34 | sortAfterUpdate: true, | |
35 | sorters: [ | |
36 | { | |
01031528 TL |
37 | property: 'name', |
38 | direction: 'ASC', | |
39 | }, | |
40 | ], | |
d7139140 DM |
41 | }); |
42 | ||
05a977a2 | 43 | let view_service_log = function() { |
ab34f5b2 TL |
44 | let { data: { service } } = me.getSelectionModel().getSelection()[0]; |
45 | Ext.create('Ext.window.Window', { | |
46 | title: gettext('Syslog') + ': ' + service, | |
f21840aa | 47 | modal: true, |
8093ded0 DC |
48 | width: 800, |
49 | height: 400, | |
50 | layout: 'fit', | |
f21840aa DM |
51 | items: { |
52 | xtype: 'proxmoxLogView', | |
ab34f5b2 | 53 | url: `/api2/extjs/nodes/${me.nodename}/syslog?service=${service}`, |
01031528 TL |
54 | log_select_timespan: 1, |
55 | }, | |
ab34f5b2 | 56 | autoShow: true, |
f21840aa | 57 | }); |
f21840aa DM |
58 | }; |
59 | ||
05a977a2 | 60 | let service_cmd = function(cmd) { |
ab34f5b2 | 61 | let { data: { service } } = me.getSelectionModel().getSelection()[0]; |
d7139140 | 62 | Proxmox.Utils.API2Request({ |
77ff40e4 | 63 | url: `/nodes/${me.nodename}/services/${service}/${cmd}`, |
d7139140 DM |
64 | method: 'POST', |
65 | failure: function(response, opts) { | |
66 | Ext.Msg.alert(gettext('Error'), response.htmlStatus); | |
67 | me.loading = true; | |
68 | }, | |
69 | success: function(response, opts) { | |
70 | rstore.startUpdate(); | |
ab34f5b2 TL |
71 | Ext.create('Proxmox.window.TaskProgress', { |
72 | upid: response.result.data, | |
73 | autoShow: true, | |
d7139140 | 74 | }); |
01031528 | 75 | }, |
d7139140 DM |
76 | }); |
77 | }; | |
78 | ||
05a977a2 | 79 | let start_btn = new Ext.Button({ |
d7139140 DM |
80 | text: gettext('Start'), |
81 | disabled: true, | |
01031528 | 82 | handler: function() { |
d7139140 | 83 | service_cmd("start"); |
01031528 | 84 | }, |
d7139140 DM |
85 | }); |
86 | ||
05a977a2 | 87 | let stop_btn = new Ext.Button({ |
d7139140 DM |
88 | text: gettext('Stop'), |
89 | disabled: true, | |
01031528 | 90 | handler: function() { |
d7139140 | 91 | service_cmd("stop"); |
01031528 | 92 | }, |
d7139140 DM |
93 | }); |
94 | ||
05a977a2 | 95 | let restart_btn = new Ext.Button({ |
d7139140 DM |
96 | text: gettext('Restart'), |
97 | disabled: true, | |
01031528 | 98 | handler: function() { |
2815c428 | 99 | service_cmd(me.restartCommand || "restart"); |
01031528 | 100 | }, |
d7139140 DM |
101 | }); |
102 | ||
05a977a2 | 103 | let syslog_btn = new Ext.Button({ |
ecb07824 DM |
104 | text: gettext('Syslog'), |
105 | disabled: true, | |
01031528 | 106 | handler: view_service_log, |
ecb07824 DM |
107 | }); |
108 | ||
05a977a2 TL |
109 | let set_button_status = function() { |
110 | let sm = me.getSelectionModel(); | |
111 | let rec = sm.getSelection()[0]; | |
d7139140 DM |
112 | |
113 | if (!rec) { | |
114 | start_btn.disable(); | |
115 | stop_btn.disable(); | |
116 | restart_btn.disable(); | |
ecb07824 | 117 | syslog_btn.disable(); |
d7139140 DM |
118 | return; |
119 | } | |
05a977a2 TL |
120 | let service = rec.data.service; |
121 | let state = rec.data.state; | |
b6eedbba | 122 | let unit = rec.data['unit-state']; |
d7139140 | 123 | |
ecb07824 DM |
124 | syslog_btn.enable(); |
125 | ||
05a977a2 TL |
126 | if (state === 'running') { |
127 | start_btn.disable(); | |
128 | restart_btn.enable(); | |
b6eedbba TL |
129 | } else if (unit !== undefined && (unit === 'masked' || unit === 'unknown')) { |
130 | start_btn.disable(); | |
131 | restart_btn.disable(); | |
05a977a2 TL |
132 | } else { |
133 | start_btn.enable(); | |
134 | restart_btn.disable(); | |
135 | } | |
136 | if (!me.startOnlyServices[service]) { | |
137 | if (state === 'running') { | |
d7139140 DM |
138 | stop_btn.enable(); |
139 | } else { | |
d7139140 DM |
140 | stop_btn.disable(); |
141 | } | |
05a977a2 | 142 | } |
d7139140 DM |
143 | }; |
144 | ||
145 | me.mon(store, 'refresh', set_button_status); | |
146 | ||
147 | Proxmox.Utils.monStoreErrors(me, rstore); | |
148 | ||
149 | Ext.apply(me, { | |
89de7ec5 TL |
150 | viewConfig: { |
151 | trackOver: false, | |
152 | stripeRows: false, // does not work with getRowClass() | |
153 | getRowClass: function(record, index) { | |
154 | let unitState = record.get('unit-state'); | |
155 | if (!unitState) { | |
156 | return ''; | |
157 | } | |
158 | if (unitState === 'masked') { | |
159 | return "proxmox-disabled-row"; | |
160 | } else if (unitState === 'unknown') { | |
161 | if (record.get('name') === 'syslog') { | |
162 | return "proxmox-disabled-row"; // replaced by journal on most hosts | |
163 | } | |
164 | return "proxmox-warning-row"; | |
165 | } | |
166 | return ''; | |
167 | }, | |
168 | }, | |
d7139140 DM |
169 | store: store, |
170 | stateful: false, | |
01031528 | 171 | tbar: [start_btn, stop_btn, restart_btn, syslog_btn], |
d7139140 DM |
172 | columns: [ |
173 | { | |
174 | header: gettext('Name'), | |
ab29b73d | 175 | flex: 1, |
d7139140 | 176 | sortable: true, |
01031528 | 177 | dataIndex: 'name', |
d7139140 DM |
178 | }, |
179 | { | |
180 | header: gettext('Status'), | |
181 | width: 100, | |
182 | sortable: true, | |
01031528 | 183 | dataIndex: 'state', |
6607de47 TL |
184 | renderer: (v, meta, rec) => rec.get('unit-state') === 'masked' |
185 | ? gettext('disabled') | |
186 | : v, | |
187 | }, | |
188 | { | |
189 | header: gettext('Active'), | |
190 | width: 100, | |
191 | sortable: true, | |
192 | hidden: true, | |
193 | dataIndex: 'active-state', | |
194 | }, | |
195 | { | |
196 | header: gettext('Unit'), | |
197 | width: 120, | |
198 | sortable: true, | |
199 | hidden: typeof PVE === 'object', // FIXME currently only PVE supports it | |
200 | dataIndex: 'unit-state', | |
d7139140 DM |
201 | }, |
202 | { | |
203 | header: gettext('Description'), | |
204 | renderer: Ext.String.htmlEncode, | |
205 | dataIndex: 'desc', | |
01031528 TL |
206 | flex: 2, |
207 | }, | |
d7139140 DM |
208 | ], |
209 | listeners: { | |
210 | selectionchange: set_button_status, | |
f21840aa | 211 | itemdblclick: view_service_log, |
d7139140 | 212 | activate: rstore.startUpdate, |
01031528 TL |
213 | destroy: rstore.stopUpdate, |
214 | }, | |
d7139140 DM |
215 | }); |
216 | ||
217 | me.callParent(); | |
01031528 | 218 | }, |
d7139140 | 219 | }); |