forked from pieter/gitbot
-
Notifications
You must be signed in to change notification settings - Fork 1
/
cybot.rb
executable file
·143 lines (117 loc) · 3.51 KB
/
cybot.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
#!/usr/bin/env ruby
#
# CyBot.
#
# Set the version
$version = "0.2"
$command = ?$ # Make sure the ? preceeds your character.
# Some modules we need.
require 'thread'
$: << File.join(File.dirname(__FILE__), "lib")
require 'pluginbase'
require 'configspace'
require 'logging'
#require 'irc'
# Create a log instance
$log = Logging.new
$log.puts "CyBot v#{$version} starting up..."
# =====================
# = Plugin management =
# =====================
# Root plugins. We always need these.
RootPlugins = ['plugin', 'user', 'test', 'irc']
# Provide a namespace for plugins.
module Plugins
end
# Tries to load the named plugin.
# Should reload if already loaded.
# Names are lower-case or properly cased?.
# RAISES: LoadError, PluginError.
def load_plugin(name)
# FIXME: Have a few places to look.
begin
# Try to load.
n = name.downcase
Plugins.module_eval { load("plugins/#{n}.rb") }
# Find the class.
if (klass = self.class.const_get(n.capitalize))
# Initialize it.
ins = klass.shared_instance
$plugins[ins.name] = ins
$log.puts "Core plugin '#{ins.name.capitalize}' loaded."
else
$log.puts "Error loading core plugin '#{n.capitalize}':"
$log.puts "Couldn't locate plugin class."
end
rescue Exception => e
$log.puts "Error loading core plugin '#{n.capitalize}':"
$log.puts e.message
$log.puts e.backtrace.join("\n")
end
end
# Some global maps.
$commands = {}
$hooks = {}
$plugins = {}
# Load global configuration.
begin
$config = ConfigSpace.new(ARGV[0])
rescue
exit 1
end
$config.prefix = <<EOS
# CyBot 0.2 main configuration file. Be careful when editing this file manually,
# as it is automatically saved run-time. On-line edit is recomended.
######### -------------------------------------------------- #########
######### REMEMBER TO SHUT DOWN THE BOT BEFORE YOU EDIT THIS #########
######### -------------------------------------------------- #########
######### This is because the bot writes the whole of this #########
######### file out when it is closed down. #########
######### -------------------------------------------------- #########
EOS
# Save everything.
@save_all_mutex = Mutex.new
def save_all(quiet = false)
@save_all_mutex.synchronize do
$log.puts 'Saving plugin data...' unless quiet
$plugins.each do |n,p|
begin
$log.print " #{n}... " unless quiet
p.save
$log.puts "Ok" unless quiet
rescue Exception => e
$log.puts "Failed with error: #{e.message}" unless quiet
end
end
$log.puts 'Saving global configuration...' unless quiet
$config.save
end
end
# Handler on ctrl-c at this point.
Signal.trap 'INT' do
Signal.trap 'INT', 'DEFAULT'
# \n leaves the ^C on its own line, and inserts our text on the line below
$log.puts "\nShutdown in progress. Press ctrl-c again to abort immediately."
$log.puts 'Asking IRC handlers to quit...'
IRC::Server.quit('Ctrl-c pressed on console...')
end
# Load root plugins.
$: << File.expand_path('support')
RootPlugins.each { |name| load_plugin(name) }
$plugins.each_value do |plugin|
plugin.on_startup if plugin.respond_to? :on_startup
end
# Set up timer to save data on regular intervals.
Thread.new do
loop do
sleep(60*60)
save_all(true)
end
end
# Done. Wait for quit.
$log.puts 'Startup complete!'
IRC::Server.wait
# Tell all plugins to save, and then save the main config.
$log.puts 'Preparing to exit...'
save_all
$log.puts 'All done, see you next time.'