win32-service 0.6.1-x86-mswin32-60
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/CHANGES +217 -0
- data/MANIFEST +18 -0
- data/README +44 -0
- data/doc/daemon.txt +155 -0
- data/doc/service.txt +365 -0
- data/lib/win32/daemon.so +0 -0
- data/lib/win32/service.rb +1579 -0
- data/test/tc_daemon.rb +55 -0
- data/test/tc_service.rb +260 -0
- data/test/tc_service_create.rb +83 -0
- data/test/tc_service_info.rb +170 -0
- data/test/tc_service_status.rb +108 -0
- metadata +80 -0
data/lib/win32/daemon.so
ADDED
Binary file
|
@@ -0,0 +1,1579 @@
|
|
1
|
+
require 'windows/error'
|
2
|
+
require 'windows/service'
|
3
|
+
require 'windows/file'
|
4
|
+
require 'windows/process'
|
5
|
+
require 'windows/security'
|
6
|
+
require 'windows/msvcrt/string'
|
7
|
+
require 'windows/msvcrt/buffer'
|
8
|
+
|
9
|
+
# The Win32 module serves as a namespace only.
|
10
|
+
module Win32
|
11
|
+
|
12
|
+
# The Service class encapsulates services controller actions, such as
|
13
|
+
# creating, starting, configuring or deleting services.
|
14
|
+
class Service
|
15
|
+
|
16
|
+
# This is the error typically raised if one of the Service methods
|
17
|
+
# should fail for any reason.
|
18
|
+
class Error < StandardError; end
|
19
|
+
|
20
|
+
include Windows::Error
|
21
|
+
include Windows::Service
|
22
|
+
include Windows::File
|
23
|
+
include Windows::Process
|
24
|
+
include Windows::Security
|
25
|
+
include Windows::MSVCRT::String
|
26
|
+
include Windows::MSVCRT::Buffer
|
27
|
+
extend Windows::Error
|
28
|
+
extend Windows::Service
|
29
|
+
extend Windows::File
|
30
|
+
extend Windows::Process
|
31
|
+
extend Windows::Security
|
32
|
+
extend Windows::MSVCRT::String
|
33
|
+
extend Windows::MSVCRT::Buffer
|
34
|
+
|
35
|
+
VERSION = '0.6.1'
|
36
|
+
|
37
|
+
# SCM security and access rights
|
38
|
+
|
39
|
+
# Includes STANDARD_RIGHTS_REQUIRED, in addition to all other rights
|
40
|
+
MANAGER_ALL_ACCESS = SC_MANAGER_ALL_ACCESS
|
41
|
+
|
42
|
+
# Required to call the CreateService function
|
43
|
+
MANAGER_CREATE_SERVICE = SC_MANAGER_CREATE_SERVICE
|
44
|
+
|
45
|
+
# Required to connect to the service control manager.
|
46
|
+
MANAGER_CONNECT = SC_MANAGER_CONNECT
|
47
|
+
|
48
|
+
# Required to call the EnumServicesStatusEx function to list services
|
49
|
+
MANAGER_ENUMERATE_SERVICE = SC_MANAGER_ENUMERATE_SERVICE
|
50
|
+
|
51
|
+
# Required to call the LockServiceDatabase function
|
52
|
+
MANAGER_LOCK = SC_MANAGER_LOCK
|
53
|
+
|
54
|
+
# Required to call the NotifyBootConfigStatus function
|
55
|
+
MANAGER_MODIFY_BOOT_CONFIG = SC_MANAGER_MODIFY_BOOT_CONFIG
|
56
|
+
|
57
|
+
# Required to call the QueryServiceLockStatus function
|
58
|
+
MANAGER_QUERY_LOCK_STATUS = SC_MANAGER_QUERY_LOCK_STATUS
|
59
|
+
|
60
|
+
# Includes STANDARD_RIGHTS_REQUIRED in addition to all access rights
|
61
|
+
ALL_ACCESS = SERVICE_ALL_ACCESS
|
62
|
+
|
63
|
+
# Required to call functions that configure existing services
|
64
|
+
CHANGE_CONFIG = SERVICE_CHANGE_CONFIG
|
65
|
+
|
66
|
+
# Required to enumerate all the services dependent on the service
|
67
|
+
ENUMERATE_DEPENDENTS = SERVICE_ENUMERATE_DEPENDENTS
|
68
|
+
|
69
|
+
# Required to make a service report its status immediately
|
70
|
+
INTERROGATE = SERVICE_INTERROGATE
|
71
|
+
|
72
|
+
# Required to control a service with a pause or resume
|
73
|
+
PAUSE_CONTINUE = SERVICE_PAUSE_CONTINUE
|
74
|
+
|
75
|
+
# Required to be able to gather configuration information about a service
|
76
|
+
QUERY_CONFIG = SERVICE_QUERY_CONFIG
|
77
|
+
|
78
|
+
# Required to be ask the SCM about the status of a service
|
79
|
+
QUERY_STATUS = SERVICE_QUERY_STATUS
|
80
|
+
|
81
|
+
# Required to call the StartService function to start the service.
|
82
|
+
START = SERVICE_START
|
83
|
+
|
84
|
+
# Required to call the ControlService function to stop the service.
|
85
|
+
STOP = SERVICE_STOP
|
86
|
+
|
87
|
+
# Required to call ControlService with a user defined control code
|
88
|
+
USER_DEFINED_CONTROL = SERVICE_USER_DEFINED_CONTROL
|
89
|
+
|
90
|
+
# Service Types
|
91
|
+
|
92
|
+
# Driver service
|
93
|
+
KERNEL_DRIVER = SERVICE_KERNEL_DRIVER
|
94
|
+
|
95
|
+
# File system driver service
|
96
|
+
FILE_SYSTEM_DRIVER = SERVICE_FILE_SYSTEM_DRIVER
|
97
|
+
|
98
|
+
# Service that runs in its own process
|
99
|
+
WIN32_OWN_PROCESS = SERVICE_WIN32_OWN_PROCESS
|
100
|
+
|
101
|
+
# Service that shares a process with one or more other services.
|
102
|
+
WIN32_SHARE_PROCESS = SERVICE_WIN32_SHARE_PROCESS
|
103
|
+
|
104
|
+
# The service can interact with the desktop
|
105
|
+
INTERACTIVE_PROCESS = SERVICE_INTERACTIVE_PROCESS
|
106
|
+
|
107
|
+
DRIVER = SERVICE_DRIVER
|
108
|
+
TYPE_ALL = SERVICE_TYPE_ALL
|
109
|
+
|
110
|
+
# Service start options
|
111
|
+
|
112
|
+
# A service started automatically by the SCM during system startup
|
113
|
+
BOOT_START = SERVICE_BOOT_START
|
114
|
+
|
115
|
+
# A device driver started by the IoInitSystem function. Drivers only
|
116
|
+
SYSTEM_START = SERVICE_SYSTEM_START
|
117
|
+
|
118
|
+
# A service started automatically by the SCM during system startup
|
119
|
+
AUTO_START = SERVICE_AUTO_START
|
120
|
+
|
121
|
+
# A service started by the SCM when a process calls StartService
|
122
|
+
DEMAND_START = SERVICE_DEMAND_START
|
123
|
+
|
124
|
+
# A service that cannot be started
|
125
|
+
DISABLED = SERVICE_DISABLED
|
126
|
+
|
127
|
+
# Error control
|
128
|
+
|
129
|
+
# Error logged, startup continues
|
130
|
+
ERROR_IGNORE = SERVICE_ERROR_IGNORE
|
131
|
+
|
132
|
+
# Error logged, pop up message, startup continues
|
133
|
+
ERROR_NORMAL = SERVICE_ERROR_NORMAL
|
134
|
+
|
135
|
+
# Error logged, startup continues, system restarted last known good config
|
136
|
+
ERROR_SEVERE = SERVICE_ERROR_SEVERE
|
137
|
+
|
138
|
+
# Error logged, startup fails, system restarted last known good config
|
139
|
+
ERROR_CRITICAL = SERVICE_ERROR_CRITICAL
|
140
|
+
|
141
|
+
# Current state
|
142
|
+
|
143
|
+
# Service is not running
|
144
|
+
STOPPED = SERVICE_STOPPED
|
145
|
+
|
146
|
+
# Service has received a start signal but is not yet running
|
147
|
+
START_PENDING = SERVICE_START_PENDING
|
148
|
+
|
149
|
+
# Service has received a stop signal but is not yet stopped
|
150
|
+
STOP_PENDING = SERVICE_STOP_PENDING
|
151
|
+
|
152
|
+
# Service is running
|
153
|
+
RUNNING = SERVICE_RUNNING
|
154
|
+
|
155
|
+
# Service has received a signal to resume but is not yet running
|
156
|
+
CONTINUE_PENDING = SERVICE_CONTINUE_PENDING
|
157
|
+
|
158
|
+
# Service has received a signal to pause but is not yet paused
|
159
|
+
PAUSE_PENDING = SERVICE_PAUSE_PENDING
|
160
|
+
|
161
|
+
# Service is paused
|
162
|
+
PAUSED = SERVICE_PAUSED
|
163
|
+
|
164
|
+
# Service controls
|
165
|
+
|
166
|
+
# Notifies service that it should stop
|
167
|
+
CONTROL_STOP = SERVICE_CONTROL_STOP
|
168
|
+
|
169
|
+
# Notifies service that it should pause
|
170
|
+
CONTROL_PAUSE = SERVICE_CONTROL_PAUSE
|
171
|
+
|
172
|
+
# Notifies service that it should resume
|
173
|
+
CONTROL_CONTINUE = SERVICE_CONTROL_CONTINUE
|
174
|
+
|
175
|
+
# Notifies service that it should return its current status information
|
176
|
+
CONTROL_INTERROGATE = SERVICE_CONTROL_INTERROGATE
|
177
|
+
|
178
|
+
# Notifies a service that its parameters have changed
|
179
|
+
CONTROL_PARAMCHANGE = SERVICE_CONTROL_PARAMCHANGE
|
180
|
+
|
181
|
+
# Notifies a service that there is a new component for binding
|
182
|
+
CONTROL_NETBINDADD = SERVICE_CONTROL_NETBINDADD
|
183
|
+
|
184
|
+
# Notifies a service that a component for binding has been removed
|
185
|
+
CONTROL_NETBINDREMOVE = SERVICE_CONTROL_NETBINDREMOVE
|
186
|
+
|
187
|
+
# Notifies a service that a component for binding has been enabled
|
188
|
+
CONTROL_NETBINDENABLE = SERVICE_CONTROL_NETBINDENABLE
|
189
|
+
|
190
|
+
# Notifies a service that a component for binding has been disabled
|
191
|
+
CONTROL_NETBINDDISABLE = SERVICE_CONTROL_NETBINDDISABLE
|
192
|
+
|
193
|
+
# Failure actions
|
194
|
+
|
195
|
+
# No action
|
196
|
+
ACTION_NONE = SC_ACTION_NONE
|
197
|
+
|
198
|
+
# Reboot the computer
|
199
|
+
ACTION_REBOOT = SC_ACTION_REBOOT
|
200
|
+
|
201
|
+
# Restart the service
|
202
|
+
ACTION_RESTART = SC_ACTION_RESTART
|
203
|
+
|
204
|
+
# Run a command
|
205
|
+
ACTION_RUN_COMMAND = SC_ACTION_RUN_COMMAND
|
206
|
+
|
207
|
+
# :stopdoc: #
|
208
|
+
StatusStruct = Struct.new('ServiceStatus', :service_type,
|
209
|
+
:current_state, :controls_accepted, :win32_exit_code,
|
210
|
+
:service_specific_exit_code, :check_point, :wait_hint, :interactive,
|
211
|
+
:pid, :service_flags
|
212
|
+
)
|
213
|
+
|
214
|
+
ConfigStruct = Struct.new('ServiceConfigInfo', :service_type,
|
215
|
+
:start_type, :error_control, :binary_path_name, :load_order_group,
|
216
|
+
:tag_id, :dependencies, :service_start_name, :display_name
|
217
|
+
)
|
218
|
+
|
219
|
+
ServiceStruct = Struct.new('ServiceInfo', :service_name,
|
220
|
+
:display_name, :service_type, :current_state, :controls_accepted,
|
221
|
+
:win32_exit_code, :service_specific_exit_code, :check_point,
|
222
|
+
:wait_hint, :binary_path_name, :start_type, :error_control,
|
223
|
+
:load_order_group, :tag_id, :start_name, :dependencies,
|
224
|
+
:description, :interactive, :pid, :service_flags, :reset_period,
|
225
|
+
:reboot_message, :command, :num_actions, :actions
|
226
|
+
)
|
227
|
+
|
228
|
+
# :startdoc: #
|
229
|
+
|
230
|
+
# Creates a new service with +service_name+ on +host+, or the local host
|
231
|
+
# if no host is specified. The +options+ parameter is a hash that can
|
232
|
+
# contain any of the following parameters, and their default values:
|
233
|
+
#
|
234
|
+
# * display_name => service_name
|
235
|
+
# * desired_access => Service::ALL_ACCESS
|
236
|
+
# * service_type => Service::WIN32_OWN_PROCESS |
|
237
|
+
# Service::INTERACTIVE_PROCESS
|
238
|
+
# * start_type => Service::DEMAND_START
|
239
|
+
# * error_control => Service::ERROR_NORMAL
|
240
|
+
# * binary_path_name => nil
|
241
|
+
# * load_order_group => nil
|
242
|
+
# * dependencies => nil
|
243
|
+
# * service_start_name => nil
|
244
|
+
# * password => nil
|
245
|
+
# * description => nil
|
246
|
+
# * failure_reset_period => nil,
|
247
|
+
# * failure_reboot_message => nil,
|
248
|
+
# * failure_command => nil,
|
249
|
+
# * failure_actions => nil,
|
250
|
+
# * failure_delay => 0
|
251
|
+
#
|
252
|
+
# Example:
|
253
|
+
#
|
254
|
+
# # Configure everything
|
255
|
+
# Service.new('some_service', nil,
|
256
|
+
# :service_type => Service::WIN32_OWN_PROCESS,
|
257
|
+
# :description => 'A custom service I wrote just for fun',
|
258
|
+
# :start_type => Service::AUTO_START,
|
259
|
+
# :error_control => Service::ERROR_NORMAL,
|
260
|
+
# :binary_path_name => 'C:\path\to\some_service.exe',
|
261
|
+
# :load_order_group => 'Network',
|
262
|
+
# :dependencies => ['W32Time','Schedule'],
|
263
|
+
# :service_start_name => 'SomeDomain\\User',
|
264
|
+
# :password => 'XXXXXXX',
|
265
|
+
# :display_name => 'This is some service',
|
266
|
+
# )
|
267
|
+
#
|
268
|
+
def initialize(service_name, host=nil, options={})
|
269
|
+
raise TypeError unless service_name.is_a?(String)
|
270
|
+
raise TypeError if host && !host.is_a?(String)
|
271
|
+
|
272
|
+
unless options.is_a?(Hash)
|
273
|
+
raise ArgumentError, 'options parameter must be a hash'
|
274
|
+
end
|
275
|
+
|
276
|
+
if options.empty?
|
277
|
+
raise ArgumentError, 'no options provided'
|
278
|
+
end
|
279
|
+
|
280
|
+
opts = {
|
281
|
+
'display_name' => nil,
|
282
|
+
'desired_access' => SERVICE_ALL_ACCESS,
|
283
|
+
'service_type' => SERVICE_WIN32_OWN_PROCESS |
|
284
|
+
SERVICE_INTERACTIVE_PROCESS,
|
285
|
+
'start_type' => SERVICE_DEMAND_START,
|
286
|
+
'error_control' => SERVICE_ERROR_NORMAL,
|
287
|
+
'binary_path_name' => nil,
|
288
|
+
'load_order_group' => nil,
|
289
|
+
'dependencies' => nil,
|
290
|
+
'service_start_name' => nil,
|
291
|
+
'password' => nil,
|
292
|
+
'description' => nil,
|
293
|
+
'failure_reset_period' => nil,
|
294
|
+
'failure_reboot_message' => nil,
|
295
|
+
'failure_command' => nil,
|
296
|
+
'failure_actions' => nil,
|
297
|
+
'failure_delay' => 0
|
298
|
+
}
|
299
|
+
|
300
|
+
# Validate the hash options
|
301
|
+
options.each{ |key, value|
|
302
|
+
key = key.to_s.downcase
|
303
|
+
unless opts.include?(key)
|
304
|
+
raise ArgumentError, "Invalid option '#{key}'"
|
305
|
+
end
|
306
|
+
opts[key] = value
|
307
|
+
}
|
308
|
+
|
309
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CREATE_SERVICE)
|
310
|
+
|
311
|
+
if handle_scm == 0
|
312
|
+
raise Error, get_last_error
|
313
|
+
end
|
314
|
+
|
315
|
+
# Display name defaults to service_name
|
316
|
+
opts['display_name'] ||= service_name
|
317
|
+
|
318
|
+
dependencies = opts['dependencies']
|
319
|
+
|
320
|
+
if dependencies && !dependencies.empty?
|
321
|
+
unless dependencies.is_a?(Array) || dependencies.is_a?(String)
|
322
|
+
raise TypeError, 'dependencies must be a string or array'
|
323
|
+
end
|
324
|
+
|
325
|
+
if dependencies.is_a?(Array)
|
326
|
+
dependencies = dependencies.join("\000")
|
327
|
+
end
|
328
|
+
|
329
|
+
dependencies += "\000"
|
330
|
+
end
|
331
|
+
|
332
|
+
handle_scs = CreateService(
|
333
|
+
handle_scm,
|
334
|
+
service_name,
|
335
|
+
opts['display_name'],
|
336
|
+
opts['desired_access'],
|
337
|
+
opts['service_type'],
|
338
|
+
opts['start_type'],
|
339
|
+
opts['error_control'],
|
340
|
+
opts['binary_path_name'],
|
341
|
+
opts['load_order_group'],
|
342
|
+
0,
|
343
|
+
dependencies,
|
344
|
+
opts['service_start_name'],
|
345
|
+
opts['password']
|
346
|
+
)
|
347
|
+
|
348
|
+
if handle_scs == 0
|
349
|
+
error = get_last_error
|
350
|
+
CloseServiceHandle(handle_scm)
|
351
|
+
CloseServiceHandle(handle_scs)
|
352
|
+
raise Error, error
|
353
|
+
end
|
354
|
+
|
355
|
+
if opts['description']
|
356
|
+
description = 0.chr * 4 # sizeof(SERVICE_DESCRIPTION)
|
357
|
+
description[0,4] = [opts['description']].pack('p*')
|
358
|
+
|
359
|
+
bool = ChangeServiceConfig2(
|
360
|
+
handle_scs,
|
361
|
+
SERVICE_CONFIG_DESCRIPTION,
|
362
|
+
description
|
363
|
+
)
|
364
|
+
|
365
|
+
unless bool
|
366
|
+
error = get_last_error
|
367
|
+
CloseServiceHandle(handle_scs)
|
368
|
+
raise Error, error
|
369
|
+
end
|
370
|
+
end
|
371
|
+
|
372
|
+
if opts['failure_reset_period'] || opts['failure_reboot_message'] ||
|
373
|
+
opts['failure_command'] || opts['failure_actions']
|
374
|
+
then
|
375
|
+
Service.configure_failure_actions(handle_scs, opts)
|
376
|
+
end
|
377
|
+
|
378
|
+
CloseServiceHandle(handle_scs)
|
379
|
+
CloseServiceHandle(handle_scm)
|
380
|
+
|
381
|
+
self
|
382
|
+
end
|
383
|
+
|
384
|
+
class << self
|
385
|
+
alias create new
|
386
|
+
end
|
387
|
+
|
388
|
+
# Configures the named +service+ on +host+, or the local host if no host
|
389
|
+
# is specified. The +options+ parameter is a hash that can contain any
|
390
|
+
# of the following parameters:
|
391
|
+
#
|
392
|
+
# * service_type
|
393
|
+
# * start_type
|
394
|
+
# * error_control
|
395
|
+
# * binary_path_name
|
396
|
+
# * load_order_group
|
397
|
+
# * dependencies
|
398
|
+
# * service_start_name
|
399
|
+
# * password (used with service_start_name)
|
400
|
+
# * display_name
|
401
|
+
# * description
|
402
|
+
# * failure_reset_period
|
403
|
+
# * failure_reboot_message
|
404
|
+
# * failure_command
|
405
|
+
# * failure_actions
|
406
|
+
# * failure_delay
|
407
|
+
#
|
408
|
+
# Examples:
|
409
|
+
#
|
410
|
+
# # Configure only the display name
|
411
|
+
# Service.configure('some_service', nil, :display_name => 'Test 33')
|
412
|
+
#
|
413
|
+
# # Configure everything
|
414
|
+
# Service.configure('some_service', nil,
|
415
|
+
# :service_type => Service::WIN32_OWN_PROCESS,
|
416
|
+
# :start_type => Service::AUTO_START,
|
417
|
+
# :error_control => Service::ERROR_NORMAL,
|
418
|
+
# :binary_path_name => 'C:\path\to\some_service.exe',
|
419
|
+
# :load_order_group => 'Network',
|
420
|
+
# :dependencies => ['W32Time','Schedule']
|
421
|
+
# :service_start_name => 'SomeDomain\\User',
|
422
|
+
# :password => 'XXXXXXX',
|
423
|
+
# :display_name => 'This is some service',
|
424
|
+
# :description => 'A custom service I wrote just for fun'
|
425
|
+
# )
|
426
|
+
#
|
427
|
+
def self.configure(service, host=nil, options={})
|
428
|
+
raise TypeError unless service.is_a?(String)
|
429
|
+
|
430
|
+
unless options.is_a?(Hash)
|
431
|
+
raise ArgumentError, 'options parameter must be a hash'
|
432
|
+
end
|
433
|
+
|
434
|
+
if options.empty?
|
435
|
+
raise ArgumentError, 'no options provided'
|
436
|
+
end
|
437
|
+
|
438
|
+
opts = {
|
439
|
+
'service_type' => SERVICE_NO_CHANGE,
|
440
|
+
'start_type' => SERVICE_NO_CHANGE,
|
441
|
+
'error_control' => SERVICE_NO_CHANGE,
|
442
|
+
'binary_path_name' => nil,
|
443
|
+
'load_order_group' => nil,
|
444
|
+
'dependencies' => nil,
|
445
|
+
'service_start_name' => nil,
|
446
|
+
'password' => nil,
|
447
|
+
'display_name' => nil,
|
448
|
+
'description' => nil,
|
449
|
+
'failure_reset_period' => nil,
|
450
|
+
'failure_reboot_message' => nil,
|
451
|
+
'failure_command' => nil,
|
452
|
+
'failure_actions' => nil,
|
453
|
+
'failure_delay' => 0
|
454
|
+
}
|
455
|
+
|
456
|
+
# Validate the hash options
|
457
|
+
options.each{ |key, value|
|
458
|
+
key = key.to_s.downcase
|
459
|
+
unless opts.include?(key)
|
460
|
+
raise ArgumentError, "Invalid option '#{key}'"
|
461
|
+
end
|
462
|
+
opts[key] = value
|
463
|
+
}
|
464
|
+
|
465
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CONNECT)
|
466
|
+
|
467
|
+
if handle_scm == 0
|
468
|
+
raise Error, get_last_error
|
469
|
+
end
|
470
|
+
|
471
|
+
desired_access = SERVICE_CHANGE_CONFIG
|
472
|
+
|
473
|
+
if opts['failure_actions']
|
474
|
+
desired_access |= SERVICE_START
|
475
|
+
end
|
476
|
+
|
477
|
+
handle_scs = OpenService(
|
478
|
+
handle_scm,
|
479
|
+
service,
|
480
|
+
desired_access
|
481
|
+
)
|
482
|
+
|
483
|
+
if handle_scs == 0
|
484
|
+
error = get_last_error
|
485
|
+
CloseServiceHandle(handle_scm)
|
486
|
+
raise Error, error
|
487
|
+
end
|
488
|
+
|
489
|
+
dependencies = opts['dependencies']
|
490
|
+
|
491
|
+
if dependencies && !dependencies.empty?
|
492
|
+
unless dependencies.is_a?(Array) || dependencies.is_a?(String)
|
493
|
+
raise TypeError, 'dependencies must be a string or array'
|
494
|
+
end
|
495
|
+
|
496
|
+
if dependencies.is_a?(Array)
|
497
|
+
dependencies = dependencies.join("\000")
|
498
|
+
end
|
499
|
+
|
500
|
+
dependencies += "\000"
|
501
|
+
end
|
502
|
+
|
503
|
+
bool = ChangeServiceConfig(
|
504
|
+
handle_scs,
|
505
|
+
opts['service_type'],
|
506
|
+
opts['start_type'],
|
507
|
+
opts['error_control'],
|
508
|
+
opts['binary_path_name'],
|
509
|
+
opts['load_order_group'],
|
510
|
+
0,
|
511
|
+
dependencies,
|
512
|
+
opts['service_start_name'],
|
513
|
+
opts['password'],
|
514
|
+
opts['display_name']
|
515
|
+
)
|
516
|
+
|
517
|
+
unless bool
|
518
|
+
error = get_last_error
|
519
|
+
CloseServiceHandle(handle_scm)
|
520
|
+
CloseServiceHandle(handle_scs)
|
521
|
+
raise Error, error
|
522
|
+
end
|
523
|
+
|
524
|
+
if opts['description']
|
525
|
+
description = 0.chr * 4 # sizeof(SERVICE_DESCRIPTION)
|
526
|
+
description[0,4] = [opts['description']].pack('p*')
|
527
|
+
|
528
|
+
bool = ChangeServiceConfig2(
|
529
|
+
handle_scs,
|
530
|
+
SERVICE_CONFIG_DESCRIPTION,
|
531
|
+
description
|
532
|
+
)
|
533
|
+
|
534
|
+
unless bool
|
535
|
+
error = get_last_error
|
536
|
+
CloseServiceHandle(handle_scs)
|
537
|
+
raise Error, error
|
538
|
+
end
|
539
|
+
end
|
540
|
+
|
541
|
+
if opts['failure_reset_period'] || opts['failure_reboot_message'] ||
|
542
|
+
opts['failure_command'] || opts['failure_actions']
|
543
|
+
then
|
544
|
+
configure_failure_actions(handle_scs, opts)
|
545
|
+
end
|
546
|
+
|
547
|
+
CloseServiceHandle(handle_scs)
|
548
|
+
CloseServiceHandle(handle_scm)
|
549
|
+
|
550
|
+
self
|
551
|
+
end
|
552
|
+
|
553
|
+
# Returns whether or not +service+ exists on +host+ or localhost, if
|
554
|
+
# no host is specified.
|
555
|
+
#
|
556
|
+
# Example:
|
557
|
+
#
|
558
|
+
# Service.exists?('W32Time') => true
|
559
|
+
#
|
560
|
+
def self.exists?(service, host=nil)
|
561
|
+
bool = false
|
562
|
+
|
563
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_ENUMERATE_SERVICE)
|
564
|
+
|
565
|
+
if handle_scm == 0
|
566
|
+
raise Error, get_last_error
|
567
|
+
end
|
568
|
+
|
569
|
+
handle_scs = OpenService(handle_scm, service, SERVICE_QUERY_STATUS)
|
570
|
+
bool = true if handle_scs > 0
|
571
|
+
|
572
|
+
CloseServiceHandle(handle_scm)
|
573
|
+
CloseServiceHandle(handle_scs)
|
574
|
+
|
575
|
+
bool
|
576
|
+
end
|
577
|
+
|
578
|
+
# Returns the display name of the specified service name, i.e. the string
|
579
|
+
# displayed in the Services GUI. Raises a Service::Error if the service
|
580
|
+
# name cannot be found.
|
581
|
+
#
|
582
|
+
# If a +host+ is provided, the information will be retrieved from that
|
583
|
+
# host. Otherwise, the information is pulled from the local host (the
|
584
|
+
# default behavior).
|
585
|
+
#
|
586
|
+
# Example:
|
587
|
+
#
|
588
|
+
# Service.get_display_name('W32Time') => 'Windows Time'
|
589
|
+
#
|
590
|
+
def self.get_display_name(service, host=nil)
|
591
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CONNECT)
|
592
|
+
|
593
|
+
if handle_scm == 0
|
594
|
+
raise Error, get_last_error
|
595
|
+
end
|
596
|
+
|
597
|
+
display_name = 0.chr * 260
|
598
|
+
display_buf = [display_name.size].pack('L')
|
599
|
+
|
600
|
+
bool = GetServiceDisplayName(
|
601
|
+
handle_scm,
|
602
|
+
service,
|
603
|
+
display_name,
|
604
|
+
display_buf
|
605
|
+
)
|
606
|
+
|
607
|
+
unless bool
|
608
|
+
error = get_last_error
|
609
|
+
CloseServiceHandle(handle_scm)
|
610
|
+
raise Error, error
|
611
|
+
end
|
612
|
+
|
613
|
+
CloseServiceHandle(handle_scm);
|
614
|
+
|
615
|
+
display_name.unpack('Z*')[0]
|
616
|
+
end
|
617
|
+
|
618
|
+
# Returns the service name of the specified service from the provided
|
619
|
+
# +display_name+. Raises a Service::Error if the +display_name+ cannote
|
620
|
+
# be found.
|
621
|
+
#
|
622
|
+
# If a +host+ is provided, the information will be retrieved from that
|
623
|
+
# host. Otherwise, the information is pulled from the local host (the
|
624
|
+
# default behavior).
|
625
|
+
#
|
626
|
+
# Example:
|
627
|
+
#
|
628
|
+
# Service.get_service_name('Windows Time') => 'W32Time'
|
629
|
+
#
|
630
|
+
def self.get_service_name(display_name, host=nil)
|
631
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CONNECT)
|
632
|
+
|
633
|
+
if handle_scm == 0
|
634
|
+
raise Error, get_last_error
|
635
|
+
end
|
636
|
+
|
637
|
+
service_name = 0.chr * 260
|
638
|
+
service_buf = [service_name.size].pack('L')
|
639
|
+
|
640
|
+
bool = GetServiceKeyName(
|
641
|
+
handle_scm,
|
642
|
+
display_name,
|
643
|
+
service_name,
|
644
|
+
service_buf
|
645
|
+
)
|
646
|
+
|
647
|
+
unless bool
|
648
|
+
error = get_last_error
|
649
|
+
CloseServiceHandle(handle_scm)
|
650
|
+
raise Error, error
|
651
|
+
end
|
652
|
+
|
653
|
+
CloseServiceHandle(handle_scm);
|
654
|
+
|
655
|
+
service_name.unpack('Z*')[0]
|
656
|
+
end
|
657
|
+
|
658
|
+
class << self
|
659
|
+
alias getdisplayname get_display_name
|
660
|
+
alias getservicename get_service_name
|
661
|
+
end
|
662
|
+
|
663
|
+
# Attempts to start the named +service+ on +host+, or the local machine
|
664
|
+
# if no host is provided. If +args+ are provided, they are passed to the
|
665
|
+
# Daemon#service_main method.
|
666
|
+
#
|
667
|
+
# Examples:
|
668
|
+
#
|
669
|
+
# # Start 'SomeSvc' on the local machine
|
670
|
+
# Service.start('SomeSvc', nil) => self
|
671
|
+
#
|
672
|
+
# # Start 'SomeSvc' on host 'foo', passing 'hello' as an argument
|
673
|
+
# Service.start('SomeSvc', 'foo', 'hello') => self
|
674
|
+
#
|
675
|
+
def self.start(service, host=nil, *args)
|
676
|
+
handle_scm = OpenSCManager(host, nil, SC_MANAGER_CONNECT)
|
677
|
+
|
678
|
+
if handle_scm == 0
|
679
|
+
raise Error, get_last_error
|
680
|
+
end
|
681
|
+
|
682
|
+
handle_scs = OpenService(handle_scm, service, SERVICE_START)
|
683
|
+
|
684
|
+
if handle_scs == 0
|
685
|
+
error = get_last_error
|
686
|
+
CloseServiceHandle(handle_scm)
|
687
|
+
raise Error, error
|
688
|
+
end
|
689
|
+
|
690
|
+
num_args = 0
|
691
|
+
|
692
|
+
if args.empty?
|
693
|
+
args = nil
|
694
|
+
else
|
695
|
+
num_args = args.length
|
696
|
+
args = args.map{ |x| [x].pack('p*') }.join
|
697
|
+
end
|
698
|
+
|
699
|
+
unless StartService(handle_scs, num_args, args)
|
700
|
+
error = get_last_error
|
701
|
+
CloseServiceHandle(handle_scs)
|
702
|
+
CloseServiceHandle(handle_scm)
|
703
|
+
raise Error, error
|
704
|
+
end
|
705
|
+
|
706
|
+
CloseServiceHandle(handle_scs)
|
707
|
+
CloseServiceHandle(handle_scm)
|
708
|
+
|
709
|
+
self
|
710
|
+
end
|
711
|
+
|
712
|
+
# Stops a the given +service+ on +host+, or the local host if no host
|
713
|
+
# is specified. Returns self.
|
714
|
+
#
|
715
|
+
# Note that attempting to stop an already stopped service raises
|
716
|
+
# Service::Error.
|
717
|
+
#
|
718
|
+
# Example:
|
719
|
+
#
|
720
|
+
# Service.stop('W32Time') => self
|
721
|
+
#
|
722
|
+
def self.stop(service, host=nil)
|
723
|
+
service_signal = SERVICE_STOP
|
724
|
+
control_signal = SERVICE_CONTROL_STOP
|
725
|
+
send_signal(service, host, service_signal, control_signal)
|
726
|
+
self
|
727
|
+
end
|
728
|
+
|
729
|
+
# Pauses the given +service+ on +host+, or the local host if no host
|
730
|
+
# is specified. Returns self
|
731
|
+
#
|
732
|
+
# Note that pausing a service that is already paused will have
|
733
|
+
# no effect and it will not raise an error.
|
734
|
+
#
|
735
|
+
# Be aware that not all services are configured to accept a pause
|
736
|
+
# command. Attempting to pause a service that isn't setup to receive
|
737
|
+
# a pause command will raise an error.
|
738
|
+
#
|
739
|
+
# Example:
|
740
|
+
#
|
741
|
+
# Service.pause('Schedule') => self
|
742
|
+
#
|
743
|
+
def self.pause(service, host=nil)
|
744
|
+
service_signal = SERVICE_PAUSE_CONTINUE
|
745
|
+
control_signal = SERVICE_CONTROL_PAUSE
|
746
|
+
send_signal(service, host, service_signal, control_signal)
|
747
|
+
self
|
748
|
+
end
|
749
|
+
|
750
|
+
# Resume the given +service+ on +host+, or the local host if no host
|
751
|
+
# is specified. Returns self.
|
752
|
+
#
|
753
|
+
# Note that resuming a service that's already running will have no
|
754
|
+
# effect and it will not raise an error.
|
755
|
+
#
|
756
|
+
# Example:
|
757
|
+
#
|
758
|
+
# Service.resume('Schedule') => self
|
759
|
+
#
|
760
|
+
def self.resume(service, host=nil)
|
761
|
+
service_signal = SERVICE_PAUSE_CONTINUE
|
762
|
+
control_signal = SERVICE_CONTROL_CONTINUE
|
763
|
+
send_signal(service, host, service_signal, control_signal)
|
764
|
+
self
|
765
|
+
end
|
766
|
+
|
767
|
+
# Deletes the specified +service+ from +host+, or the local host if
|
768
|
+
# no host is specified. Returns self.
|
769
|
+
#
|
770
|
+
# Technical note. This method is not instantaneous. The service is first
|
771
|
+
# marked for deletion from the service control manager database. Then all
|
772
|
+
# handles to the service are closed. Then an attempt to stop the service
|
773
|
+
# is made. If the service cannot be stopped, the service control manager
|
774
|
+
# database entry is removed when the system is restarted.
|
775
|
+
#
|
776
|
+
# Example:
|
777
|
+
#
|
778
|
+
# Service.delete('SomeService') => self
|
779
|
+
#
|
780
|
+
def self.delete(service, host=nil)
|
781
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CREATE_SERVICE)
|
782
|
+
|
783
|
+
if handle_scm == 0
|
784
|
+
raise Error, get_last_error
|
785
|
+
end
|
786
|
+
|
787
|
+
handle_scs = OpenService(handle_scm, service, DELETE)
|
788
|
+
|
789
|
+
if handle_scs == 0
|
790
|
+
error = get_last_error
|
791
|
+
CloseServiceHandle(handle_scm)
|
792
|
+
raise Error, error
|
793
|
+
end
|
794
|
+
|
795
|
+
unless DeleteService(handle_scs)
|
796
|
+
error = get_last_error
|
797
|
+
CloseServiceHandle(handle_scm)
|
798
|
+
CloseServiceHandle(handle_scs)
|
799
|
+
raise Error, error
|
800
|
+
end
|
801
|
+
|
802
|
+
CloseServiceHandle(handle_scm)
|
803
|
+
CloseServiceHandle(handle_scs)
|
804
|
+
|
805
|
+
self
|
806
|
+
end
|
807
|
+
|
808
|
+
# Returns a ServiceConfigInfo struct containing the configuration
|
809
|
+
# information about +service+ on +host+, or the local host if no
|
810
|
+
# host is specified.
|
811
|
+
#
|
812
|
+
# Example:
|
813
|
+
#
|
814
|
+
# Service.config_info('W32Time') => <struct ServiceConfigInfo ...>
|
815
|
+
#--
|
816
|
+
# This contains less information that the ServiceInfo struct that
|
817
|
+
# is returned with the Service.services method, but is faster for
|
818
|
+
# looking up basic information for a single service.
|
819
|
+
#
|
820
|
+
def self.config_info(service, host=nil)
|
821
|
+
raise TypeError if host && !host.is_a?(String)
|
822
|
+
|
823
|
+
handle_scm = OpenSCManager(host, nil, SC_MANAGER_ENUMERATE_SERVICE)
|
824
|
+
|
825
|
+
if handle_scm == 0
|
826
|
+
raise Error, get_last_error
|
827
|
+
end
|
828
|
+
|
829
|
+
handle_scs = OpenService(handle_scm, service, SERVICE_QUERY_CONFIG)
|
830
|
+
|
831
|
+
if handle_scs == 0
|
832
|
+
error = get_last_error
|
833
|
+
CloseServiceHandle(handle_scm)
|
834
|
+
raise Error, error
|
835
|
+
end
|
836
|
+
|
837
|
+
# First, get the buf size needed
|
838
|
+
bytes_needed = [0].pack('L')
|
839
|
+
|
840
|
+
bool = QueryServiceConfig(handle_scs, nil, 0, bytes_needed)
|
841
|
+
|
842
|
+
if !bool && GetLastError() != ERROR_INSUFFICIENT_BUFFER
|
843
|
+
error = get_last_error
|
844
|
+
CloseServiceHandle(handle_scs)
|
845
|
+
CloseServiceHandle(handle_scm)
|
846
|
+
raise Error, error
|
847
|
+
end
|
848
|
+
|
849
|
+
buf = 0.chr * bytes_needed.unpack('L')[0]
|
850
|
+
bytes = [0].pack('L')
|
851
|
+
|
852
|
+
bool = QueryServiceConfig(handle_scs, buf, buf.size, bytes_needed)
|
853
|
+
|
854
|
+
unless bool
|
855
|
+
error = get_last_error
|
856
|
+
CloseServiceHandle(handle_scs)
|
857
|
+
CloseServiceHandle(handle_scm)
|
858
|
+
raise Error, error
|
859
|
+
end
|
860
|
+
|
861
|
+
CloseServiceHandle(handle_scs)
|
862
|
+
CloseServiceHandle(handle_scm)
|
863
|
+
|
864
|
+
binary_path_name = 0.chr * 260
|
865
|
+
load_order_group = 0.chr * 260
|
866
|
+
dependencies = 0.chr * 260
|
867
|
+
service_start_name = 0.chr * 260
|
868
|
+
display_name = 0.chr * 260
|
869
|
+
|
870
|
+
strcpy(binary_path_name, buf[12,4].unpack('L')[0])
|
871
|
+
binary_path_name = binary_path_name.unpack('Z*')[0]
|
872
|
+
|
873
|
+
strcpy(load_order_group, buf[16,4].unpack('L')[0])
|
874
|
+
load_order_group = load_order_group.unpack('Z*')[0]
|
875
|
+
|
876
|
+
dependencies = get_dependencies(buf[24,4].unpack('L').first)
|
877
|
+
|
878
|
+
strcpy(service_start_name, buf[28,4].unpack('L')[0])
|
879
|
+
service_start_name = service_start_name.unpack('Z*')[0]
|
880
|
+
|
881
|
+
strcpy(display_name, buf[32,4].unpack('L')[0])
|
882
|
+
display_name = display_name.unpack('Z*')[0]
|
883
|
+
|
884
|
+
ConfigStruct.new(
|
885
|
+
get_service_type(buf[0,4].unpack('L')[0]),
|
886
|
+
get_start_type(buf[4,4].unpack('L')[0]),
|
887
|
+
get_error_control(buf[8,4].unpack('L')[0]),
|
888
|
+
binary_path_name,
|
889
|
+
load_order_group,
|
890
|
+
buf[20,4].unpack('L')[0],
|
891
|
+
dependencies,
|
892
|
+
service_start_name,
|
893
|
+
display_name
|
894
|
+
)
|
895
|
+
end
|
896
|
+
|
897
|
+
# Returns a ServiceStatus struct indicating the status of service +name+
|
898
|
+
# on +host+, or the localhost if none is provided.
|
899
|
+
#
|
900
|
+
# Example:
|
901
|
+
#
|
902
|
+
# Service.status('W32Time') => <struct Struct::ServiceStatus ...>
|
903
|
+
#
|
904
|
+
def self.status(service, host=nil)
|
905
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_ENUMERATE_SERVICE)
|
906
|
+
|
907
|
+
if handle_scm == 0
|
908
|
+
raise Error, get_last_error
|
909
|
+
end
|
910
|
+
|
911
|
+
handle_scs = OpenService(
|
912
|
+
handle_scm,
|
913
|
+
service,
|
914
|
+
SERVICE_QUERY_STATUS
|
915
|
+
)
|
916
|
+
|
917
|
+
if handle_scs == 0
|
918
|
+
error = get_last_error
|
919
|
+
CloseServiceHandle(handle_scm)
|
920
|
+
raise Error, error
|
921
|
+
end
|
922
|
+
|
923
|
+
# SERVICE_STATUS_PROCESS struct
|
924
|
+
status = [0,0,0,0,0,0,0,0,0].pack('LLLLLLLLL')
|
925
|
+
bytes = [0].pack('L')
|
926
|
+
|
927
|
+
bool = QueryServiceStatusEx(
|
928
|
+
handle_scs,
|
929
|
+
SC_STATUS_PROCESS_INFO,
|
930
|
+
status,
|
931
|
+
status.size,
|
932
|
+
bytes
|
933
|
+
)
|
934
|
+
|
935
|
+
unless bool
|
936
|
+
raise Error, get_last_error
|
937
|
+
end
|
938
|
+
|
939
|
+
dw_service_type = status[0,4].unpack('L').first
|
940
|
+
|
941
|
+
service_type = get_service_type(dw_service_type)
|
942
|
+
current_state = get_current_state(status[4,4].unpack('L').first)
|
943
|
+
controls = get_controls_accepted(status[8,4].unpack('L').first)
|
944
|
+
interactive = dw_service_type & SERVICE_INTERACTIVE_PROCESS > 0
|
945
|
+
|
946
|
+
# Note that the pid and service flags will always return 0 if you're
|
947
|
+
# on Windows NT 4 or using a version of Ruby compiled with VC++ 6
|
948
|
+
# or earlier.
|
949
|
+
#
|
950
|
+
status_struct = StatusStruct.new(
|
951
|
+
service_type,
|
952
|
+
current_state,
|
953
|
+
controls,
|
954
|
+
status[12,4].unpack('L').first, # Win32ExitCode
|
955
|
+
status[16,4].unpack('L').first, # ServiceSpecificExitCode
|
956
|
+
status[20,4].unpack('L').first, # CheckPoint
|
957
|
+
status[24,4].unpack('L').first, # WaitHint
|
958
|
+
interactive,
|
959
|
+
status[28,4].unpack('L').first, # ProcessId
|
960
|
+
status[32,4].unpack('L').first # ServiceFlags
|
961
|
+
)
|
962
|
+
|
963
|
+
CloseServiceHandle(handle_scs)
|
964
|
+
CloseServiceHandle(handle_scm)
|
965
|
+
|
966
|
+
status_struct
|
967
|
+
end
|
968
|
+
|
969
|
+
# Enumerates over a list of service types on +host+, or the local
|
970
|
+
# machine if no host is specified, yielding a ServiceInfo struct for
|
971
|
+
# each service.
|
972
|
+
#
|
973
|
+
# If a +group+ is specified, then only those services that belong to
|
974
|
+
# that load order group are enumerated. If an empty string is provided,
|
975
|
+
# then only services that do not belong to any group are enumerated. If
|
976
|
+
# this parameter is nil (the default), group membership is ignored and
|
977
|
+
# all services are enumerated. This value is not case sensitive.
|
978
|
+
#
|
979
|
+
# Examples:
|
980
|
+
#
|
981
|
+
# # Enumerate over all services on the localhost
|
982
|
+
# Service.services{ |service| p service }
|
983
|
+
#
|
984
|
+
# # Enumerate over all services on a remote host
|
985
|
+
# Service.services('some_host'){ |service| p service }
|
986
|
+
#
|
987
|
+
# # Enumerate over all 'network' services locally
|
988
|
+
# Service.services(nil, 'network'){ |service| p service }
|
989
|
+
#
|
990
|
+
def self.services(host=nil, group=nil)
|
991
|
+
unless host.nil?
|
992
|
+
raise TypeError unless host.is_a?(String) # Avoid strange errors
|
993
|
+
end
|
994
|
+
|
995
|
+
unless group.nil?
|
996
|
+
raise TypeError unless group.is_a?(String) # Avoid strange errors
|
997
|
+
end
|
998
|
+
|
999
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_ENUMERATE_SERVICE)
|
1000
|
+
|
1001
|
+
if handle_scm == 0
|
1002
|
+
raise Error, get_last_error
|
1003
|
+
end
|
1004
|
+
|
1005
|
+
bytes_needed = [0].pack('L')
|
1006
|
+
services_returned = [0].pack('L')
|
1007
|
+
resume_handle = [0].pack('L')
|
1008
|
+
|
1009
|
+
# The first call is used to determine the required buffer size
|
1010
|
+
bool = EnumServicesStatusEx(
|
1011
|
+
handle_scm,
|
1012
|
+
SC_ENUM_PROCESS_INFO,
|
1013
|
+
SERVICE_WIN32 | SERVICE_DRIVER,
|
1014
|
+
SERVICE_STATE_ALL,
|
1015
|
+
0,
|
1016
|
+
0,
|
1017
|
+
bytes_needed,
|
1018
|
+
services_returned,
|
1019
|
+
resume_handle,
|
1020
|
+
group
|
1021
|
+
)
|
1022
|
+
|
1023
|
+
err_num = GetLastError()
|
1024
|
+
|
1025
|
+
if !bool && err_num == ERROR_MORE_DATA
|
1026
|
+
service_buf = 0.chr * bytes_needed.unpack('L').first
|
1027
|
+
else
|
1028
|
+
error = get_last_error(err_num)
|
1029
|
+
CloseServiceHandle(handle_scm)
|
1030
|
+
raise Error, error
|
1031
|
+
end
|
1032
|
+
|
1033
|
+
bool = EnumServicesStatusEx(
|
1034
|
+
handle_scm,
|
1035
|
+
SC_ENUM_PROCESS_INFO,
|
1036
|
+
SERVICE_WIN32 | SERVICE_DRIVER,
|
1037
|
+
SERVICE_STATE_ALL,
|
1038
|
+
service_buf,
|
1039
|
+
service_buf.size,
|
1040
|
+
bytes_needed,
|
1041
|
+
services_returned,
|
1042
|
+
resume_handle,
|
1043
|
+
group
|
1044
|
+
)
|
1045
|
+
|
1046
|
+
unless bool
|
1047
|
+
error = get_last_error
|
1048
|
+
CloseServiceHandle(handle_scm)
|
1049
|
+
raise Error, error
|
1050
|
+
end
|
1051
|
+
|
1052
|
+
num_services = services_returned.unpack('L').first
|
1053
|
+
|
1054
|
+
index = 0
|
1055
|
+
services_array = [] unless block_given?
|
1056
|
+
|
1057
|
+
1.upto(num_services){ |num|
|
1058
|
+
service_name = 0.chr * 260
|
1059
|
+
display_name = 0.chr * 260
|
1060
|
+
|
1061
|
+
info = service_buf[index, 44] # sizeof(SERVICE_STATUS_PROCESS)
|
1062
|
+
|
1063
|
+
strcpy(service_name, info[0,4].unpack('L').first)
|
1064
|
+
strcpy(display_name, info[4,4].unpack('L').first)
|
1065
|
+
|
1066
|
+
service_name = service_name.unpack('Z*')[0]
|
1067
|
+
display_name = display_name.unpack('Z*')[0]
|
1068
|
+
|
1069
|
+
dw_service_type = info[8,4].unpack('L').first
|
1070
|
+
|
1071
|
+
service_type = get_service_type(dw_service_type)
|
1072
|
+
current_state = get_current_state(info[12,4].unpack('L').first)
|
1073
|
+
controls = get_controls_accepted(info[16,4].unpack('L').first)
|
1074
|
+
interactive = dw_service_type & SERVICE_INTERACTIVE_PROCESS > 0
|
1075
|
+
win_exit_code = info[20,4].unpack('L').first
|
1076
|
+
ser_exit_code = info[24,4].unpack('L').first
|
1077
|
+
check_point = info[28,4].unpack('L').first
|
1078
|
+
wait_hint = info[32,4].unpack('L').first
|
1079
|
+
pid = info[36,4].unpack('L').first
|
1080
|
+
service_flags = info[40,4].unpack('L').first
|
1081
|
+
|
1082
|
+
handle_scs = OpenService(
|
1083
|
+
handle_scm,
|
1084
|
+
service_name,
|
1085
|
+
SERVICE_QUERY_CONFIG
|
1086
|
+
)
|
1087
|
+
|
1088
|
+
if handle_scs == 0
|
1089
|
+
error = get_last_error
|
1090
|
+
CloseServiceHandle(handle_scm)
|
1091
|
+
CloseServiceHandle(handle_scs)
|
1092
|
+
raise Error, error
|
1093
|
+
end
|
1094
|
+
|
1095
|
+
config_buf = get_config_info(handle_scs)
|
1096
|
+
|
1097
|
+
if config_buf != ERROR_FILE_NOT_FOUND
|
1098
|
+
binary_path = 0.chr * 260
|
1099
|
+
strcpy(binary_path, config_buf[12,4].unpack('L').first)
|
1100
|
+
binary_path = binary_path.unpack('Z*')[0]
|
1101
|
+
|
1102
|
+
load_order = 0.chr * 260
|
1103
|
+
strcpy(load_order, config_buf[16,4].unpack('L').first)
|
1104
|
+
load_order = load_order.unpack('Z*')[0]
|
1105
|
+
|
1106
|
+
start_name = 0.chr * 260
|
1107
|
+
strcpy(start_name, config_buf[28,4].unpack('L').first)
|
1108
|
+
start_name = start_name.unpack('Z*')[0]
|
1109
|
+
|
1110
|
+
start_type = get_start_type(config_buf[4,4].unpack('L').first)
|
1111
|
+
error_ctrl = get_error_control(config_buf[8,4].unpack('L').first)
|
1112
|
+
|
1113
|
+
tag_id = config_buf[20,4].unpack('L').first
|
1114
|
+
|
1115
|
+
deps = get_dependencies(config_buf[24,4].unpack('L').first)
|
1116
|
+
|
1117
|
+
description = 0.chr * 1024
|
1118
|
+
buf = get_config2_info(handle_scs, SERVICE_CONFIG_DESCRIPTION)
|
1119
|
+
|
1120
|
+
strcpy(description, buf[0,4].unpack('L').first)
|
1121
|
+
description = description.unpack('Z*')[0]
|
1122
|
+
else
|
1123
|
+
msg = "WARNING: The registry entry for the #{service_name} "
|
1124
|
+
msg += "service could not be found."
|
1125
|
+
warn msg
|
1126
|
+
|
1127
|
+
binary_path = nil
|
1128
|
+
load_order = nil
|
1129
|
+
start_name = nil
|
1130
|
+
start_type = nil
|
1131
|
+
error_ctrl = nil
|
1132
|
+
tag_id = nil
|
1133
|
+
deps = nil
|
1134
|
+
description = nil
|
1135
|
+
end
|
1136
|
+
|
1137
|
+
buf2 = get_config2_info(handle_scs, SERVICE_CONFIG_FAILURE_ACTIONS)
|
1138
|
+
|
1139
|
+
if buf2 != ERROR_FILE_NOT_FOUND
|
1140
|
+
reset_period = buf2[0,4].unpack('L').first
|
1141
|
+
|
1142
|
+
reboot_msg = 0.chr * 260
|
1143
|
+
strcpy(reboot_msg, buf2[4,4].unpack('L').first)
|
1144
|
+
reboot_msg = reboot_msg.unpack('Z*')[0]
|
1145
|
+
|
1146
|
+
command = 0.chr * 260
|
1147
|
+
strcpy(command, buf2[8,4].unpack('L').first)
|
1148
|
+
command = command.unpack('Z*')[0]
|
1149
|
+
|
1150
|
+
num_actions = buf2[12,4].unpack('L').first
|
1151
|
+
actions = nil
|
1152
|
+
|
1153
|
+
if num_actions > 0
|
1154
|
+
action_ptr = buf2[16,4].unpack('L').first
|
1155
|
+
action_buf = [0,0].pack('LL') * num_actions
|
1156
|
+
memcpy(action_buf, action_ptr, action_buf.size)
|
1157
|
+
|
1158
|
+
i = 0
|
1159
|
+
actions = {}
|
1160
|
+
num_actions.times{ |n|
|
1161
|
+
action_type, delay = action_buf[i, 8].unpack('LL')
|
1162
|
+
action_type = get_action_type(action_type)
|
1163
|
+
actions[n+1] = {:action_type => action_type, :delay => delay}
|
1164
|
+
i += 8
|
1165
|
+
}
|
1166
|
+
end
|
1167
|
+
else
|
1168
|
+
reset_period = nil
|
1169
|
+
reboot_message = nil
|
1170
|
+
command = nil
|
1171
|
+
actions = nil
|
1172
|
+
end
|
1173
|
+
|
1174
|
+
CloseServiceHandle(handle_scs)
|
1175
|
+
|
1176
|
+
struct = ServiceStruct.new(
|
1177
|
+
service_name,
|
1178
|
+
display_name,
|
1179
|
+
service_type,
|
1180
|
+
current_state,
|
1181
|
+
controls,
|
1182
|
+
win_exit_code,
|
1183
|
+
ser_exit_code,
|
1184
|
+
check_point,
|
1185
|
+
wait_hint,
|
1186
|
+
binary_path,
|
1187
|
+
start_type,
|
1188
|
+
error_ctrl,
|
1189
|
+
load_order,
|
1190
|
+
tag_id,
|
1191
|
+
start_name,
|
1192
|
+
deps,
|
1193
|
+
description,
|
1194
|
+
interactive,
|
1195
|
+
pid,
|
1196
|
+
service_flags,
|
1197
|
+
reset_period,
|
1198
|
+
reboot_msg,
|
1199
|
+
command,
|
1200
|
+
num_actions,
|
1201
|
+
actions
|
1202
|
+
)
|
1203
|
+
|
1204
|
+
if block_given?
|
1205
|
+
yield struct
|
1206
|
+
else
|
1207
|
+
services_array << struct
|
1208
|
+
end
|
1209
|
+
|
1210
|
+
index += 44 # sizeof(SERVICE_STATUS_PROCESS)
|
1211
|
+
}
|
1212
|
+
|
1213
|
+
CloseServiceHandle(handle_scm)
|
1214
|
+
|
1215
|
+
block_given? ? nil : services_array
|
1216
|
+
end
|
1217
|
+
|
1218
|
+
private
|
1219
|
+
|
1220
|
+
# Configures failure actions for a given service.
|
1221
|
+
#
|
1222
|
+
def self.configure_failure_actions(handle_scs, opts)
|
1223
|
+
if opts['failure_actions']
|
1224
|
+
token_handle = 0.chr * 4
|
1225
|
+
|
1226
|
+
bool = OpenProcessToken(
|
1227
|
+
GetCurrentProcess(),
|
1228
|
+
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
|
1229
|
+
token_handle
|
1230
|
+
)
|
1231
|
+
|
1232
|
+
unless bool
|
1233
|
+
error = get_last_error
|
1234
|
+
CloseServiceHandle(handle_scs)
|
1235
|
+
raise Error, error
|
1236
|
+
end
|
1237
|
+
|
1238
|
+
token_handle = token_handle.unpack('L').first
|
1239
|
+
|
1240
|
+
# Get the LUID for shutdown privilege.
|
1241
|
+
luid = 0.chr * 8
|
1242
|
+
|
1243
|
+
unless LookupPrivilegeValue('', 'SeShutdownPrivilege', luid)
|
1244
|
+
error = get_last_error
|
1245
|
+
CloseServiceHandle(handle_scs)
|
1246
|
+
raise Error, error
|
1247
|
+
end
|
1248
|
+
|
1249
|
+
tkp = [1].pack('L') + luid + [SE_PRIVILEGE_ENABLED].pack('L')
|
1250
|
+
|
1251
|
+
# Enable shutdown privilege in access token of this process
|
1252
|
+
bool = AdjustTokenPrivileges(
|
1253
|
+
token_handle,
|
1254
|
+
0,
|
1255
|
+
tkp,
|
1256
|
+
tkp.size,
|
1257
|
+
nil,
|
1258
|
+
nil
|
1259
|
+
)
|
1260
|
+
|
1261
|
+
unless bool
|
1262
|
+
error = get_last_error
|
1263
|
+
CloseServiceHandle(handle_scs)
|
1264
|
+
raise Error, error
|
1265
|
+
end
|
1266
|
+
end
|
1267
|
+
|
1268
|
+
fail_buf = 0.chr * 20 # sizeof(SERVICE_FAILURE_ACTIONS)
|
1269
|
+
|
1270
|
+
if opts['failure_reset_period']
|
1271
|
+
fail_buf[0,4] = [opts['failure_reset_period']].pack('L')
|
1272
|
+
end
|
1273
|
+
|
1274
|
+
if opts['failure_reboot_message']
|
1275
|
+
fail_buf[4,4] = [opts['failure_reboot_message']].pack('p*')
|
1276
|
+
end
|
1277
|
+
|
1278
|
+
if opts['failure_command']
|
1279
|
+
fail_buf[8,4] = [opts['failure_command']].pack('p*')
|
1280
|
+
end
|
1281
|
+
|
1282
|
+
if opts['failure_actions']
|
1283
|
+
actions = []
|
1284
|
+
|
1285
|
+
opts['failure_actions'].each{ |action|
|
1286
|
+
action_buf = 0.chr * 8
|
1287
|
+
action_buf[0, 4] = [action].pack('L')
|
1288
|
+
action_buf[4, 4] = [opts['failure_delay']].pack('L')
|
1289
|
+
actions << action_buf
|
1290
|
+
}
|
1291
|
+
|
1292
|
+
actions = actions.join
|
1293
|
+
|
1294
|
+
fail_buf[12,4] = [opts['failure_actions'].length].pack('L')
|
1295
|
+
fail_buf[16,4] = [actions].pack('p*')
|
1296
|
+
end
|
1297
|
+
|
1298
|
+
bool = ChangeServiceConfig2(
|
1299
|
+
handle_scs,
|
1300
|
+
SERVICE_CONFIG_FAILURE_ACTIONS,
|
1301
|
+
fail_buf
|
1302
|
+
)
|
1303
|
+
|
1304
|
+
unless bool
|
1305
|
+
error = get_last_error
|
1306
|
+
CloseServiceHandle(handle_scs)
|
1307
|
+
raise Error, error
|
1308
|
+
end
|
1309
|
+
end
|
1310
|
+
|
1311
|
+
# Unravels a pointer to an array of dependencies. Takes the address
|
1312
|
+
# that points the array as an argument.
|
1313
|
+
#
|
1314
|
+
def self.get_dependencies(address)
|
1315
|
+
dep_buf = ""
|
1316
|
+
|
1317
|
+
while address != 0
|
1318
|
+
char_buf = 0.chr
|
1319
|
+
memcpy(char_buf, address, 1)
|
1320
|
+
address += 1
|
1321
|
+
dep_buf += char_buf
|
1322
|
+
break if dep_buf[-2,2] == "\0\0"
|
1323
|
+
end
|
1324
|
+
|
1325
|
+
dependencies = []
|
1326
|
+
|
1327
|
+
if dep_buf != "\0\0"
|
1328
|
+
dependencies = dep_buf.split("\000\000").first.split(0.chr)
|
1329
|
+
end
|
1330
|
+
|
1331
|
+
dependencies
|
1332
|
+
end
|
1333
|
+
|
1334
|
+
# Returns a human readable string indicating the action type.
|
1335
|
+
#
|
1336
|
+
def self.get_action_type(action_type)
|
1337
|
+
case action_type
|
1338
|
+
when SC_ACTION_NONE
|
1339
|
+
'none'
|
1340
|
+
when SC_ACTION_REBOOT
|
1341
|
+
'reboot'
|
1342
|
+
when SC_ACTION_RESTART
|
1343
|
+
'restart'
|
1344
|
+
when SC_ACTION_RUN_COMMAND
|
1345
|
+
'command'
|
1346
|
+
else
|
1347
|
+
'unknown'
|
1348
|
+
end
|
1349
|
+
end
|
1350
|
+
|
1351
|
+
# Shortcut for QueryServiceConfig. Returns the buffer. In rare cases
|
1352
|
+
# the underlying registry entry may have been deleted, but the service
|
1353
|
+
# still exists. In that case, the ERROR_FILE_NOT_FOUND value is returned
|
1354
|
+
# instead.
|
1355
|
+
#
|
1356
|
+
def self.get_config_info(handle)
|
1357
|
+
bytes_needed = [0].pack('L')
|
1358
|
+
|
1359
|
+
# First attempt at QueryServiceConfig is to get size needed
|
1360
|
+
bool = QueryServiceConfig(handle, 0, 0, bytes_needed)
|
1361
|
+
|
1362
|
+
err_num = GetLastError()
|
1363
|
+
|
1364
|
+
if !bool && err_num == ERROR_INSUFFICIENT_BUFFER
|
1365
|
+
config_buf = 0.chr * bytes_needed.unpack('L').first
|
1366
|
+
elsif err_num == ERROR_FILE_NOT_FOUND
|
1367
|
+
return err_num
|
1368
|
+
else
|
1369
|
+
error = get_last_error(err_num)
|
1370
|
+
CloseServiceHandle(handle)
|
1371
|
+
raise Error, error
|
1372
|
+
end
|
1373
|
+
|
1374
|
+
bytes_needed = [0].pack('L')
|
1375
|
+
|
1376
|
+
# Second attempt at QueryServiceConfig gets the actual info
|
1377
|
+
bool = QueryServiceConfig(
|
1378
|
+
handle,
|
1379
|
+
config_buf,
|
1380
|
+
config_buf.size,
|
1381
|
+
bytes_needed
|
1382
|
+
)
|
1383
|
+
|
1384
|
+
unless bool
|
1385
|
+
error = get_last_error
|
1386
|
+
CloseServiceHandle(handle)
|
1387
|
+
raise Error, error
|
1388
|
+
end
|
1389
|
+
|
1390
|
+
config_buf
|
1391
|
+
end
|
1392
|
+
|
1393
|
+
# Shortcut for QueryServiceConfig2. Returns the buffer.
|
1394
|
+
#
|
1395
|
+
def self.get_config2_info(handle, info_level)
|
1396
|
+
bytes_needed = [0].pack('L')
|
1397
|
+
|
1398
|
+
# First attempt at QueryServiceConfig2 is to get size needed
|
1399
|
+
bool = QueryServiceConfig2(handle, info_level, 0, 0, bytes_needed)
|
1400
|
+
|
1401
|
+
err_num = GetLastError()
|
1402
|
+
|
1403
|
+
if !bool && err_num == ERROR_INSUFFICIENT_BUFFER
|
1404
|
+
config2_buf = 0.chr * bytes_needed.unpack('L').first
|
1405
|
+
elsif err_num == ERROR_FILE_NOT_FOUND
|
1406
|
+
return err_num
|
1407
|
+
else
|
1408
|
+
CloseServiceHandle(handle)
|
1409
|
+
raise Error, get_last_error(err_num)
|
1410
|
+
end
|
1411
|
+
|
1412
|
+
bytes_needed = [0].pack('L')
|
1413
|
+
|
1414
|
+
# Second attempt at QueryServiceConfig2 gets the actual info
|
1415
|
+
bool = QueryServiceConfig2(
|
1416
|
+
handle,
|
1417
|
+
info_level,
|
1418
|
+
config2_buf,
|
1419
|
+
config2_buf.size,
|
1420
|
+
bytes_needed
|
1421
|
+
)
|
1422
|
+
|
1423
|
+
unless bool
|
1424
|
+
error = GetLastError()
|
1425
|
+
CloseServiceHandle(handle)
|
1426
|
+
raise Error, get_last_error(error)
|
1427
|
+
end
|
1428
|
+
|
1429
|
+
config2_buf
|
1430
|
+
end
|
1431
|
+
|
1432
|
+
# Returns a human readable string indicating the error control
|
1433
|
+
#
|
1434
|
+
def self.get_error_control(error_control)
|
1435
|
+
case error_control
|
1436
|
+
when SERVICE_ERROR_CRITICAL
|
1437
|
+
'critical'
|
1438
|
+
when SERVICE_ERROR_IGNORE
|
1439
|
+
'ignore'
|
1440
|
+
when SERVICE_ERROR_NORMAL
|
1441
|
+
'normal'
|
1442
|
+
when SERVICE_ERROR_SEVERE
|
1443
|
+
'severe'
|
1444
|
+
else
|
1445
|
+
nil
|
1446
|
+
end
|
1447
|
+
end
|
1448
|
+
|
1449
|
+
# Returns a human readable string indicating the start type.
|
1450
|
+
#
|
1451
|
+
def self.get_start_type(start_type)
|
1452
|
+
case start_type
|
1453
|
+
when SERVICE_AUTO_START
|
1454
|
+
'auto start'
|
1455
|
+
when SERVICE_BOOT_START
|
1456
|
+
'boot start'
|
1457
|
+
when SERVICE_DEMAND_START
|
1458
|
+
'demand start'
|
1459
|
+
when SERVICE_DISABLED
|
1460
|
+
'disabled'
|
1461
|
+
when SERVICE_SYSTEM_START
|
1462
|
+
'system start'
|
1463
|
+
else
|
1464
|
+
nil
|
1465
|
+
end
|
1466
|
+
end
|
1467
|
+
|
1468
|
+
# Returns an array of human readable strings indicating the controls
|
1469
|
+
# that the service accepts.
|
1470
|
+
#
|
1471
|
+
def self.get_controls_accepted(controls)
|
1472
|
+
array = []
|
1473
|
+
case controls
|
1474
|
+
when controls & SERVICE_ACCEPT_NETBINDCHANGE > 0
|
1475
|
+
array.push('netbind change')
|
1476
|
+
when controls & SERVICE_ACCEPT_PARAMCHANGE > 0
|
1477
|
+
array.push('param change')
|
1478
|
+
when controls & SERVICE_PAUSE_CONTINUE > 0
|
1479
|
+
array.push('pause continue')
|
1480
|
+
when controls & SERVICE_ACCEPT_SHUTDOWN > 0
|
1481
|
+
array.push('shutdown')
|
1482
|
+
when controls & SERVICE_ACCEPT_STOP > 0
|
1483
|
+
array.push('stop')
|
1484
|
+
when controls & SERVICE_ACCEPT_HARDWAREPROFILECHANGE > 0
|
1485
|
+
array.push('hardware profile change')
|
1486
|
+
when controls & SERVICE_ACCEPT_POWEREVENT > 0
|
1487
|
+
array.push('power event')
|
1488
|
+
when controls & SERVICE_ACCEPT_SESSIONCHANGE > 0
|
1489
|
+
array.push('session change')
|
1490
|
+
end
|
1491
|
+
|
1492
|
+
array.empty? ? nil : array
|
1493
|
+
end
|
1494
|
+
|
1495
|
+
# Converts a service state numeric constant into a readable string.
|
1496
|
+
#
|
1497
|
+
def self.get_current_state(state)
|
1498
|
+
case state
|
1499
|
+
when SERVICE_CONTINUE_PENDING
|
1500
|
+
'continue pending'
|
1501
|
+
when SERVICE_PAUSE_PENDING
|
1502
|
+
'pause pending'
|
1503
|
+
when SERVICE_PAUSED
|
1504
|
+
'paused'
|
1505
|
+
when SERVICE_RUNNING
|
1506
|
+
'running'
|
1507
|
+
when SERVICE_START_PENDING
|
1508
|
+
'start pending'
|
1509
|
+
when SERVICE_STOP_PENDING
|
1510
|
+
'stop pending'
|
1511
|
+
when SERVICE_STOPPED
|
1512
|
+
'stopped'
|
1513
|
+
else
|
1514
|
+
nil
|
1515
|
+
end
|
1516
|
+
end
|
1517
|
+
|
1518
|
+
# Converts a service type numeric constant into a human readable string.
|
1519
|
+
#
|
1520
|
+
def self.get_service_type(service_type)
|
1521
|
+
case service_type
|
1522
|
+
when SERVICE_FILE_SYSTEM_DRIVER
|
1523
|
+
'file system driver'
|
1524
|
+
when SERVICE_KERNEL_DRIVER
|
1525
|
+
'kernel driver'
|
1526
|
+
when SERVICE_WIN32_OWN_PROCESS
|
1527
|
+
'own process'
|
1528
|
+
when SERVICE_WIN32_SHARE_PROCESS
|
1529
|
+
'share process'
|
1530
|
+
when SERVICE_RECOGNIZER_DRIVER
|
1531
|
+
'recognizer driver'
|
1532
|
+
when SERVICE_DRIVER
|
1533
|
+
'driver'
|
1534
|
+
when SERVICE_WIN32
|
1535
|
+
'win32'
|
1536
|
+
when SERVICE_TYPE_ALL
|
1537
|
+
'all'
|
1538
|
+
when SERVICE_INTERACTIVE_PROCESS | SERVICE_WIN32_OWN_PROCESS
|
1539
|
+
'own process, interactive'
|
1540
|
+
when SERVICE_INTERACTIVE_PROCESS | SERVICE_WIN32_SHARE_PROCESS
|
1541
|
+
'share process, interactive'
|
1542
|
+
else
|
1543
|
+
nil
|
1544
|
+
end
|
1545
|
+
end
|
1546
|
+
|
1547
|
+
# A shortcut method that simplifies the various service control methods.
|
1548
|
+
#
|
1549
|
+
def self.send_signal(service, host, service_signal, control_signal)
|
1550
|
+
handle_scm = OpenSCManager(host, 0, SC_MANAGER_CONNECT)
|
1551
|
+
|
1552
|
+
if handle_scm == 0
|
1553
|
+
raise Error, get_last_error
|
1554
|
+
end
|
1555
|
+
|
1556
|
+
handle_scs = OpenService(handle_scm, service, service_signal)
|
1557
|
+
|
1558
|
+
if handle_scs == 0
|
1559
|
+
error = get_last_error
|
1560
|
+
CloseServiceHandle(handle_scm)
|
1561
|
+
raise Error, error
|
1562
|
+
end
|
1563
|
+
|
1564
|
+
status = [0,0,0,0,0,0,0].pack('LLLLLLL')
|
1565
|
+
|
1566
|
+
unless ControlService(handle_scs, control_signal, status)
|
1567
|
+
error = get_last_error
|
1568
|
+
CloseServiceHandle(handle_scs)
|
1569
|
+
CloseServiceHandle(handle_scm)
|
1570
|
+
raise Error, error
|
1571
|
+
end
|
1572
|
+
|
1573
|
+
CloseServiceHandle(handle_scs)
|
1574
|
+
CloseServiceHandle(handle_scm)
|
1575
|
+
|
1576
|
+
status
|
1577
|
+
end
|
1578
|
+
end
|
1579
|
+
end
|