forked from ottobehrens/gemstone-scripts
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstone.rb
276 lines (226 loc) · 7.79 KB
/
stone.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
require File.join(File.dirname(__FILE__), 'gemstone_installation')
require File.join(File.dirname(__FILE__), 'topaz')
require 'date'
require 'fileutils'
class Stone
attr_accessor :username, :password
attr_reader :name
attr_reader :log_directory
attr_reader :data_directory
attr_reader :backup_directory
attr_reader :extent_name
def Stone.existing(name, gemstone_installation=GemStoneInstallation.current)
fail "Stone does not exist" if not gemstone_installation.stones.include? name
new(name, gemstone_installation)
end
def Stone.create(name, gemstone_installation=GemStoneInstallation.current)
fail "Cannot create stone #{name}: the conf file already exists in #{gemstone_installation.config_directory}" if gemstone_installation.stones.include? name
instance = new(name, gemstone_installation)
instance.initialize_new_stone
instance
end
def initialize(name, gemstone_installation=GemStoneInstallation.current, username="DataCurator", password="swordfish")
@name = name
@username = username
@password = password
@log_directory = "#{gemstone_installation.base_log_directory}/#@name"
@data_directory = "#{gemstone_installation.installation_extent_directory}/#@name"
@backup_directory = gemstone_installation.backup_directory
@extent_name = gemstone_installation.initial_extent_name
@gemstone_installation = gemstone_installation
initialize_gemstone_environment
end
def initialize_gemstone_environment
@gemstone_installation.set_gemstone_installation_environment
ENV['GEMSTONE_LOGDIR'] = log_directory
ENV['GEMSTONE_DATADIR'] = data_directory
end
# Bare bones stone with nothing loaded, specialise for your situation
def initialize_new_stone
create_skeleton
copy_clean_extent
end
def create_skeleton
create_config_file
if !File.exists?(extent_directory) then FileUtils.mkdir_p extent_directory end
if !File.exists?(log_directory) then FileUtils.mkdir_p log_directory end
tranlog_directories.each do | tranlog_dir |
if !File.exists?(tranlog_dir) then FileUtils.mkdir_p tranlog_dir end
end
end
# Will remove everything in the stone's data directory!
def destroy!
fail "Can not destroy a running stone" if running?
FileUtils.rm_rf extent_filename
tranlog_directories.each do | tranlog_dir |
if tranlog_dir != '/dev/null' then FileUtils.rm_rf "#{tranlog_dir}/*" end
end
end
def recreate!
stop_system
destroy!
initialize_new_stone
start
end
def running?(wait_time = -1)
gs_sh "waitstone #@name #{wait_time} 1>/dev/null"
end
def status
if running?
gs_sh "gslist -clv #@name"
else
puts "GemStone server \"#@name\" not running."
end
end
def start_system
start
end
def stop_system
stop
end
def start
# Startstone can use single or double quotes around the stone name, so check for either (Yucch)
gs_sh "startstone -z #{system_config_filename} -l #{File.join(log_directory, @name)}.log #{@name} | grep Info]:.*[\\\'\\\"]#{@name}"
running?(10)
self
end
def stop
if running?
gs_sh "stopstone -i #{name} #{username} #{password} 1>/dev/null"
else
puts "GemStone server \"#@name\" not running."
end
self
end
def restart
stop
start
end
def restart_system
stop_system
start_system
end
def copy_to(copy_stone)
fail "You can not copy a running stone" if running?
fail "You can not copy to a stone that is running" if copy_stone.running?
copy_stone.create_skeleton
FileUtils.install(extent_filename, copy_stone.extent_filename, :mode => 0660)
end
def self.tranlog_number_from(string)
((/\[.*SmallInteger\] (-?\d*)/.match(string))[1]).to_i
end
def log_sh(command_line)
log_command_line(command_line)
system(redirect_command_line_to_logfile(command_line))
end
def full_backup(file_name_prefix=backup_filename_prefix_for_today)
result = run_topaz_command("System startCheckpointSync")
fail "Could not start checkpoint, got #{result[-2].last}" if /\[.*Boolean\] true/ !~ result[-2].last.last
result = run_topaz_command("SystemRepository startNewLog")
tranlog_number = Stone.tranlog_number_from(result[-2].last.last)
fail "Could not start a new tranlog" if tranlog_number == -1
run_topaz_commands("System abortTransaction", "SystemRepository fullBackupCompressedTo: '#{file_name_prefix}'")
end
def restore_latest_full_backup
restore_full_backup_from_named_file
end
def restore_full_backup(stone_name, for_date=Date.today)
restore_full_backup_from_named_file(backup_filename(stone_name, for_date))
end
def restore_full_backup_from_named_file(file_name=backup_filename_for_today)
recreate!
run_topaz_commands("System abortTransaction. SystemRepository restoreFromBackup: '#{file_name}'")
run_topaz_commands("SystemRepository commitRestore")
end
def backup_filename_for_today
backup_filename(name, Date.today)
end
def backup_filename_prefix_for_today
backup_filename_prefix(name, Date.today)
end
def backup_filename(for_stone_name, for_date)
"#{backup_filename_prefix(for_stone_name, for_date)}.gz"
end
def backup_filename_prefix(for_stone_name, for_date)
"#{backup_directory}/#{for_stone_name}_#{for_date.strftime('%F')}.full"
end
def input_file(topaz_script_filename, login_first=true)
topaz_commands(["input #{topaz_script_filename}", "commit"], login_first)
end
def system_config_filename
"#{@gemstone_installation.config_directory}/#@name.conf"
end
def create_config_file
if not File.exists?(system_config_filename) then
require 'erb'
File.open(system_config_filename, "w") do | file |
file.write(ERB.new(config_file_template).result(binding))
end
end
self
end
def config_file_template
File.open(File.dirname(__FILE__) + "/stone.conf.template").read
end
def extent_directory
File.join(data_directory, "extent")
end
def extent_filename
File.join(extent_directory, extent_name)
end
def scratch_directory
File.join(data_directory, "scratch.")
end
def tranlog_directories
directory = File.join(data_directory, "tranlog")
[directory, directory]
end
def topaz_logfile
"#{log_directory}/topaz.log"
end
def command_logfile
FileUtils.mkdir_p log_directory
"#{log_directory}/stone_command_output.log"
end
def gemstone_installation_directory
@gemstone_installation.installation_directory
end
def run_topaz_command(command)
run_topaz_commands(command)
end
def run_topaz_commands(*commands)
topaz_commands(["run", commands.join(". "), "%"])
end
def gs_sh(command_line, &block)
initialize_gemstone_environment
@gemstone_installation.gs_sh(command_line, &block)
end
def log_gs_sh(command_line)
log_command_line(command_line)
gs_sh redirect_command_line_to_logfile(command_line)
end
def copy_clean_extent
FileUtils.install(@gemstone_installation.initial_extent, extent_filename, :mode => 0660)
end
def copy_seaside_extent
FileUtils.install(@gemstone_installation.seaside_extent, extent_filename, :mode => 0660)
end
def topaz_commands(user_commands, login_first=true)
commands = ["set u #{username} p #{password} gemstone #{name}" ]
commands << "login" if login_first
commands.push(["limit oops 100",
"limit bytes 1000",
"display oops",
"iferr 1 stack",
"iferr 2 exit 3"],
user_commands)
Topaz.new(self).commands(commands, topaz_logfile)
end
private
def log_command_line(command_line)
File.open(command_logfile, "a") { |file| file.puts "SHELL_CMD #{Date.today.strftime('%F %T')}: #{command_line}" }
end
def redirect_command_line_to_logfile(command_line)
"#{command_line} 2>&1 >> #{command_logfile}"
end
end