arbi 1.0.2
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/bin/arbi +7 -0
- data/bin/arbid +13 -0
- data/lib/arbi.rb +218 -0
- data/lib/arbi/common.rb +3 -0
- data/lib/arbi/plugins/batteries.rb +85 -0
- data/lib/arbi/plugins/cpu.rb +89 -0
- data/lib/arbi/plugins/diskspace.rb +68 -0
- data/lib/arbi/plugins/net.rb +120 -0
- data/lib/arbi/plugins/ram.rb +29 -0
- data/lib/arbi/plugins/thermal.rb +65 -0
- metadata +104 -0
data/bin/arbi
ADDED
data/bin/arbid
ADDED
data/lib/arbi.rb
ADDED
@@ -0,0 +1,218 @@
|
|
1
|
+
#!/usr/bin/ruby
|
2
|
+
|
3
|
+
class Regexp
|
4
|
+
alias oldto_s to_s
|
5
|
+
def to_s
|
6
|
+
regex = oldto_s.gsub(/^\(\?|\)/, '')
|
7
|
+
"/#{regex.split(/:/, 2)[1]}/#{regex.split(':')[0].split('-')[0]}"
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
module Arbi
|
12
|
+
|
13
|
+
require 'getopt/long'
|
14
|
+
include Getopt
|
15
|
+
|
16
|
+
VERSION = "1.0.2"
|
17
|
+
|
18
|
+
@cmd = []
|
19
|
+
|
20
|
+
def self.add_plugin regex, instance
|
21
|
+
@cmd += [[regex, instance]]
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.cmd
|
25
|
+
@cmd
|
26
|
+
end
|
27
|
+
|
28
|
+
def self.show_version
|
29
|
+
puts "Arbi (Client|Server) v#{VERSION}"
|
30
|
+
exit 0
|
31
|
+
end
|
32
|
+
|
33
|
+
class Server
|
34
|
+
require 'socket'
|
35
|
+
|
36
|
+
def initialize
|
37
|
+
@threads = Array.new
|
38
|
+
@sessions = Array.new
|
39
|
+
@address = 'localhost'
|
40
|
+
@port = 6969
|
41
|
+
parse_args
|
42
|
+
@server = TCPServer.new(@address, @port)
|
43
|
+
end
|
44
|
+
|
45
|
+
def show_help
|
46
|
+
puts "Arbi Server, USAGE:"
|
47
|
+
puts "\t#{$0} [switches]"
|
48
|
+
puts "\t\t--bind-address|-a\tAddress to bind, default to \"127.0.0.1\""
|
49
|
+
puts "\t\t --port|-p\tPort to use for server, default to 40"
|
50
|
+
puts "\t\t --version|-v\tPrint version and exit"
|
51
|
+
puts "\t\t --help|-h\tPrint this help and exit"
|
52
|
+
exit 0
|
53
|
+
end
|
54
|
+
|
55
|
+
def start
|
56
|
+
@servert = Thread.new do
|
57
|
+
while(session = @server.accept)
|
58
|
+
new_client session
|
59
|
+
end
|
60
|
+
end
|
61
|
+
@servert.join
|
62
|
+
end
|
63
|
+
|
64
|
+
def close
|
65
|
+
@sessions.each{ |session|
|
66
|
+
session.close
|
67
|
+
}
|
68
|
+
|
69
|
+
@threads.each{ |thread|
|
70
|
+
thread.kill
|
71
|
+
}
|
72
|
+
@server.close
|
73
|
+
@servert.kill
|
74
|
+
end
|
75
|
+
|
76
|
+
private
|
77
|
+
|
78
|
+
def parse_args
|
79
|
+
begin
|
80
|
+
opts = Arbi::Long.getopts(
|
81
|
+
["--bind-address", "-a", Arbi::REQUIRED],
|
82
|
+
["--port", "-p", Arbi::REQUIRED],
|
83
|
+
["--help", "-h", Arbi::BOOLEAN],
|
84
|
+
["--version", "-v", Arbi::BOOLEAN]
|
85
|
+
)
|
86
|
+
rescue Getopt::Long::Error => e
|
87
|
+
puts "Arguments error: #{e}"
|
88
|
+
exit 0
|
89
|
+
end
|
90
|
+
|
91
|
+
@address = opts['a'] if opts['a']
|
92
|
+
@port = opts['p'] if opts['p']
|
93
|
+
show_help if opts['h']
|
94
|
+
Arbi::show_version if opts['v']
|
95
|
+
end
|
96
|
+
|
97
|
+
def new_client session
|
98
|
+
@threads.push Thread.start(session){ |session|
|
99
|
+
while message = session.gets
|
100
|
+
toggle = true
|
101
|
+
message = message.strip
|
102
|
+
break if message =~ /^QUIT$/i
|
103
|
+
|
104
|
+
if message =~ /^HELP$/i
|
105
|
+
session.print "help:\r\n"
|
106
|
+
Arbi::cmd.each { |pair|
|
107
|
+
session.print "#{pair[0]}\r\n"
|
108
|
+
}
|
109
|
+
session.print "/^quit$/i\r\n"
|
110
|
+
session.print "/^version$/i\r\n"
|
111
|
+
session.print "/^help$/i\r\n"
|
112
|
+
session.print "END\r\n"
|
113
|
+
toggle = !toggle
|
114
|
+
end
|
115
|
+
|
116
|
+
if message =~ /^VERSION$/i
|
117
|
+
session.print "version:\r\nArbi (Client|Server) v#{VERSION}\r\nEND\r\n"
|
118
|
+
toggle = !toggle
|
119
|
+
end
|
120
|
+
|
121
|
+
Arbi::cmd.each { |pair|
|
122
|
+
if message =~ pair[0]
|
123
|
+
session.print (pair[1].class == Class ? pair[1].protocolize(pair[1].get_infos) : pair[1].class.protocolize(pair[1].get_infos))
|
124
|
+
toggle = !toggle
|
125
|
+
end
|
126
|
+
}
|
127
|
+
session.print "error:\r\nCommand doesn't exist\r\nEND\r\n" if toggle
|
128
|
+
end
|
129
|
+
@sessions -= [session]
|
130
|
+
session.close
|
131
|
+
}
|
132
|
+
@sessions.push session
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
class Client
|
137
|
+
def initialize
|
138
|
+
@address = 'localhost'
|
139
|
+
@port = 6969
|
140
|
+
@command = "help\r\nquit\r\n"
|
141
|
+
parse_args
|
142
|
+
@sock = TCPSocket.new(@address, @port)
|
143
|
+
end
|
144
|
+
|
145
|
+
def start
|
146
|
+
@sock.print @command
|
147
|
+
@command = @command.strip.split(/\r\n/).map(&:upcase)
|
148
|
+
@command.pop
|
149
|
+
toggle = false
|
150
|
+
while (line = @sock.gets)
|
151
|
+
puts (command = @command.shift) + ":"
|
152
|
+
buff = ""
|
153
|
+
while (l = @sock.gets).strip !~ /^END$/i do; buff << l; end
|
154
|
+
buff.gsub!(/END\s+$/m, '')
|
155
|
+
case line.strip
|
156
|
+
when /^error:$/i
|
157
|
+
puts "ERROR: #{buff}"
|
158
|
+
next
|
159
|
+
when /^help:$/i
|
160
|
+
puts buff
|
161
|
+
when /^version$:/i
|
162
|
+
puts buff
|
163
|
+
end
|
164
|
+
Arbi::cmd.each{|cmd|
|
165
|
+
puts (cmd[1].class == Class ? cmd[1].friendlize(buff) : cmd[1].class.friendlize(buff)) if command =~ cmd[0]
|
166
|
+
}
|
167
|
+
puts if @command != []
|
168
|
+
end
|
169
|
+
end
|
170
|
+
|
171
|
+
def close
|
172
|
+
@sock.close
|
173
|
+
end
|
174
|
+
|
175
|
+
def show_help
|
176
|
+
puts "Arbi client:"
|
177
|
+
puts "\tUSAGE: #{$0} [switches]"
|
178
|
+
puts "\t\t --help|-h\tshow this helps"
|
179
|
+
puts "\t\t--version|-v\tshow version of arbi"
|
180
|
+
puts "\t\t--address|-a\tset the address to connect, default to 'localhost'"
|
181
|
+
puts "\t\t --port|-p\tset the port to connect, default to 40"
|
182
|
+
puts "\t\t--command|-c\tset commands to execute, defaults is 'help'"
|
183
|
+
exit 0
|
184
|
+
end
|
185
|
+
|
186
|
+
private
|
187
|
+
|
188
|
+
def parse_args
|
189
|
+
begin
|
190
|
+
opts = Arbi::Long.getopts(
|
191
|
+
["--help"],
|
192
|
+
["--version"],
|
193
|
+
["--address", "-a", Arbi::REQUIRED],
|
194
|
+
["--port", "-p", Arbi::REQUIRED],
|
195
|
+
["--commands", "-c", Arbi::REQUIRED]
|
196
|
+
)
|
197
|
+
rescue Arbi::Long::Error => e
|
198
|
+
p ARGV
|
199
|
+
$stderr.puts "Arguments error: #{e}"
|
200
|
+
exit 1
|
201
|
+
end
|
202
|
+
|
203
|
+
Arbi::show_version if opts["v"]
|
204
|
+
show_help if opts["h"]
|
205
|
+
@address = opts["a"] if opts["a"]
|
206
|
+
@port = opts["p"] if opts["p"]
|
207
|
+
@command = "#{opts["c"]},quit".gsub(/,+/, ',').gsub(/\s+/, '').split(/,/).
|
208
|
+
delete_if{|x|x=~/^quit$/i}.uniq.push('quit', '').join("\r\n") if opts["c"]
|
209
|
+
end
|
210
|
+
end
|
211
|
+
end
|
212
|
+
|
213
|
+
require 'arbi/plugins/batteries'
|
214
|
+
require 'arbi/plugins/cpu'
|
215
|
+
require 'arbi/plugins/diskspace'
|
216
|
+
require 'arbi/plugins/net'
|
217
|
+
require 'arbi/plugins/ram'
|
218
|
+
require 'arbi/plugins/thermal'
|
data/lib/arbi/common.rb
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
class Batteries
|
2
|
+
require 'arbi/common'
|
3
|
+
|
4
|
+
def initialize
|
5
|
+
@AC = {'sanity' => false, 'state' => false, 'percent' => false}
|
6
|
+
@batteries = Dir.glob "/proc/acpi/battery/*"
|
7
|
+
end
|
8
|
+
|
9
|
+
def get_infos
|
10
|
+
batteries = Hash.new
|
11
|
+
@batteries.each { |battery|
|
12
|
+
batteries.merge!(get_battery_infos(battery))
|
13
|
+
}
|
14
|
+
batteries.merge average batteries
|
15
|
+
end
|
16
|
+
|
17
|
+
def self.protocolize batteries
|
18
|
+
str = "batteries:\r\n"
|
19
|
+
batteries.each { |key, value|
|
20
|
+
str << "#{key}|#{value["percent"] || '-'}"
|
21
|
+
if key != "AVERAGE"
|
22
|
+
str << "|#{value["state"] || '-'}|#{value["sanity"] || '-'}"
|
23
|
+
end
|
24
|
+
str << "\r\n"
|
25
|
+
}
|
26
|
+
str << "END\r\n"
|
27
|
+
end
|
28
|
+
|
29
|
+
def self.friendlize raw
|
30
|
+
raw.strip!
|
31
|
+
max = raw.lines.map{|x| x.split(/\|/)[0].length}.max + 4
|
32
|
+
str = "NAME".ljust(max) + "PERC STATE SANITY\r\n"
|
33
|
+
raw.split(/\r?\n/).each { |line|
|
34
|
+
datas = line.split /\|/
|
35
|
+
str << datas[0].ljust(max) + datas[1].ljust(8)
|
36
|
+
if datas[0] != "AVERAGE"
|
37
|
+
str << datas[2].ljust(12) + datas[3]
|
38
|
+
end
|
39
|
+
str << "\r\n"
|
40
|
+
}
|
41
|
+
str
|
42
|
+
end
|
43
|
+
|
44
|
+
private
|
45
|
+
|
46
|
+
def average batteries
|
47
|
+
avg, n = 0, 0
|
48
|
+
batteries.each do |key, value|
|
49
|
+
if value["percent"]
|
50
|
+
avg += value["percent"].gsub(/%$/, '').to_i
|
51
|
+
n += 1
|
52
|
+
end
|
53
|
+
end
|
54
|
+
avg = (avg.to_f / n).to_s.gsub /\.0/, '' + '%'
|
55
|
+
|
56
|
+
({'AVERAGE' => {
|
57
|
+
'sanity' => '-',
|
58
|
+
'state' => '-',
|
59
|
+
'percent' => avg
|
60
|
+
}
|
61
|
+
})
|
62
|
+
end
|
63
|
+
|
64
|
+
def get_battery_infos dir
|
65
|
+
info = file_get_content dir + "/info"
|
66
|
+
state = file_get_content dir + "/state"
|
67
|
+
present = {'yes' => true, 'no' => false}[state.match(/^present:\s*?([^\s]+?)$/)[1]]
|
68
|
+
if !present then return {dir.gsub(/^\/proc\/acpi\/battery\//, '') => @AC} end
|
69
|
+
({dir.gsub(/^\/proc\/acpi\/battery\//, '') => {
|
70
|
+
'sanity' => sprintf('%.1f%%', 100.0 /
|
71
|
+
(info.match(/^design capacity:\s*?([^\s]+?) mAh$/)[1].to_i) *
|
72
|
+
(info.match(/^last full capacity:\s*?([^\s]+?) mAh$/)[1].to_i)).gsub(/\.0%$/, '%'),
|
73
|
+
'state' => state.match(/^charging state:\s*?([^\s]+?)$/)[1],
|
74
|
+
'percent' => sprintf('%.1f%%', 100.0 /
|
75
|
+
(info.match(/^last full capacity:\s*?([^\s]+?) mAh$/)[1].to_i) *
|
76
|
+
(state.match(/^remaining capacity:\s*?([^\s]+?) mAh$/)[1].to_i)).gsub(/\.0%$/, '%')
|
77
|
+
}})
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
if __FILE__ == $0
|
82
|
+
puts Batteries.friendlize Batteries.protocolize(Batteries.new.get_infos).gsub(/^batteries:\s+|^END\s+/m, '').strip
|
83
|
+
else
|
84
|
+
Arbi.add_plugin /^batteries$/i, Batteries.new
|
85
|
+
end
|
@@ -0,0 +1,89 @@
|
|
1
|
+
class Cpu
|
2
|
+
require 'arbi/common'
|
3
|
+
require 'thread'
|
4
|
+
|
5
|
+
def initialize
|
6
|
+
@prev = {:idle => Hash.new, :total => Hash.new}
|
7
|
+
@percents = Hash.new
|
8
|
+
@mutex = Mutex.new
|
9
|
+
@serv = Thread.new do
|
10
|
+
while true
|
11
|
+
@mutex.lock
|
12
|
+
update
|
13
|
+
@mutex.unlock
|
14
|
+
sleep 2
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
def get_infos
|
20
|
+
@mutex.lock
|
21
|
+
perc = @percents
|
22
|
+
@mutex.unlock
|
23
|
+
perc
|
24
|
+
end
|
25
|
+
|
26
|
+
def self.protocolize percents
|
27
|
+
str = "cpu:\r\n"
|
28
|
+
percents.each { |key, value|
|
29
|
+
str += "#{key}|#{value}\r\n"
|
30
|
+
}
|
31
|
+
str + "END\r\n"
|
32
|
+
end
|
33
|
+
|
34
|
+
def self.friendlize raw
|
35
|
+
raw = raw.to_s.strip
|
36
|
+
max = raw.lines.map{|x|x.split(/\|/)[0].length}.max + 4
|
37
|
+
str = "NAME".ljust(max) + "PERC\r\n"
|
38
|
+
raw.split(/\r?\n/).each { |line|
|
39
|
+
datas = line.split /\|/
|
40
|
+
str << datas[0].ljust(max) + datas[1] + "\r\n"
|
41
|
+
}
|
42
|
+
str
|
43
|
+
end
|
44
|
+
|
45
|
+
def close
|
46
|
+
@serv.kill
|
47
|
+
end
|
48
|
+
|
49
|
+
private
|
50
|
+
|
51
|
+
def update
|
52
|
+
file_get_content('/proc/stat').split("\n").each do |line|
|
53
|
+
if line =~ /^cpu/
|
54
|
+
datas = line.split /\s+/
|
55
|
+
name = datas.shift
|
56
|
+
name = (name == 'cpu' ? 'AVERAGE' : name)
|
57
|
+
total = eval datas.join('+')
|
58
|
+
idle = datas[3].to_i
|
59
|
+
|
60
|
+
if not @prev[:idle].include? name or not @prev[:total].include? name
|
61
|
+
@prev[:idle][name] = @prev[:total][name] = 0
|
62
|
+
end
|
63
|
+
|
64
|
+
@percents[name] = (100 * ((total - @prev[:total][name]) - (idle - @prev[:idle][name])) / (total - @prev[:total][name])).to_s + "%"
|
65
|
+
@prev[:idle][name] = idle
|
66
|
+
@prev[:total][name] = total
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
average = {"AVERAGE" => @percents["AVERAGE"]} and @percents.delete("AVERAGE") and @percents.merge!(average)
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
if __FILE__ == $0
|
75
|
+
x = Cpu.new
|
76
|
+
["INT", "KILL", "ABRT", "ILL", "QUIT"].each do |sig|
|
77
|
+
trap(sig) do
|
78
|
+
x.close
|
79
|
+
exit 0
|
80
|
+
end
|
81
|
+
end
|
82
|
+
10.times {
|
83
|
+
sleep 2
|
84
|
+
puts Cpu.friendlize Cpu.protocolize(x.get_infos).gsub(/^cpu:\s+|END\s+/m, '')
|
85
|
+
}
|
86
|
+
x.close
|
87
|
+
else
|
88
|
+
Arbi.add_plugin /^cpu$/i, Cpu.new
|
89
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
class Diskspace
|
2
|
+
require 'arbi/common'
|
3
|
+
require 'sys/filesystem'
|
4
|
+
include Sys
|
5
|
+
|
6
|
+
def get_infos
|
7
|
+
devices = Array.new
|
8
|
+
get_devices.each do |device|
|
9
|
+
begin
|
10
|
+
diskstat = Filesystem.stat device['point']
|
11
|
+
device.merge!({'usage' => (100 - (100.0 / diskstat.blocks * diskstat.blocks_available).round).to_s + "%",
|
12
|
+
'space' => diskstat.blocks * diskstat.fragment_size})
|
13
|
+
devices += [device]
|
14
|
+
rescue
|
15
|
+
end
|
16
|
+
end
|
17
|
+
devices
|
18
|
+
end
|
19
|
+
|
20
|
+
def self.protocolize devices
|
21
|
+
str = "devices:\r\n"
|
22
|
+
devices.each do |device|
|
23
|
+
str += "#{device['device']}|#{device['point']}|#{device['usage']}|#{self.unitize device['space']}\r\n"
|
24
|
+
end
|
25
|
+
str + "END\r\n"
|
26
|
+
end
|
27
|
+
|
28
|
+
def self.friendlize raw
|
29
|
+
raw = raw.to_s.strip
|
30
|
+
dmax = [raw.lines.map{|line|line.split(/\|/)[0].length}.max + 4, 8].max
|
31
|
+
mmax = [raw.lines.map{|line|line.split(/\|/)[1].length}.max + 4, 7].max
|
32
|
+
str = "DEVICE".ljust(dmax) + "MOUNT".ljust(mmax) + "USE % SPACE\r\n"
|
33
|
+
raw.split(/\r?\n/).each { |line|
|
34
|
+
datas = line.split /\|/
|
35
|
+
str << datas[0].ljust(dmax) + datas[1].ljust(mmax) + datas[2].ljust(7) + datas[3] + "\r\n"
|
36
|
+
}
|
37
|
+
str
|
38
|
+
end
|
39
|
+
|
40
|
+
private
|
41
|
+
|
42
|
+
def get_devices
|
43
|
+
devices = Array.new
|
44
|
+
Filesystem.mounts do |fs|
|
45
|
+
devices += [{'device' => fs.name, 'point' => fs.mount_point}]
|
46
|
+
end
|
47
|
+
devices
|
48
|
+
end
|
49
|
+
|
50
|
+
def self.unitize misure
|
51
|
+
u = 'b'
|
52
|
+
%w(Kb Mb Gb Tb).each do |i|
|
53
|
+
if misure >= 1024
|
54
|
+
misure /= 1024.0
|
55
|
+
u = i
|
56
|
+
else
|
57
|
+
return misure.round.to_s + u
|
58
|
+
end
|
59
|
+
end
|
60
|
+
misure.round.to_s + u
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
if __FILE__ == $0
|
65
|
+
puts Diskspace.friendlize Diskspace.protocolize(Diskspace.new.get_infos).gsub(/^devices:\s+|END\s+$/m, '')
|
66
|
+
else
|
67
|
+
Arbi::add_plugin /^diskstats$/i, Diskspace.new
|
68
|
+
end
|
@@ -0,0 +1,120 @@
|
|
1
|
+
class Net
|
2
|
+
require 'arbi/common'
|
3
|
+
require 'thread'
|
4
|
+
|
5
|
+
def initialize
|
6
|
+
@stats = Hash.new
|
7
|
+
@prev = {:up => Hash.new, :down => Hash.new}
|
8
|
+
@mutex = Mutex.new
|
9
|
+
@serv = Thread.new do
|
10
|
+
while true
|
11
|
+
@mutex.lock
|
12
|
+
update
|
13
|
+
@mutex.unlock
|
14
|
+
sleep 2
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
def get_infos
|
20
|
+
@mutex.lock
|
21
|
+
s = @stats
|
22
|
+
@mutex.unlock
|
23
|
+
s
|
24
|
+
end
|
25
|
+
|
26
|
+
def self.protocolize stats
|
27
|
+
str = "net:\r\n"
|
28
|
+
stats.each { |key, value|
|
29
|
+
str += "#{key}|#{value[:up]}|#{value[:down]}|#{value[:state] ? 'on' : 'off'}"
|
30
|
+
if value[:quality]
|
31
|
+
str += "|#{value[:essid]}|#{value[:quality]}"
|
32
|
+
end
|
33
|
+
str += "\r\n"
|
34
|
+
}
|
35
|
+
str + "END\r\n"
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.friendlize raw
|
39
|
+
raw = raw.to_s.strip
|
40
|
+
nmax = raw.lines.map{|x|x.split(/\|/)[0].length}.max + 4
|
41
|
+
umax = raw.lines.map{|x|x.split(/\|/)[1].gsub(/^(\d+\.\d).*$/, '\1').length}.max + 2
|
42
|
+
dmax = raw.lines.map{|x|x.split(/\|/)[2].gsub(/^(\d+\.\d).*$/, '\1').length}.max + 2
|
43
|
+
emax = [[raw.lines.map{|x|x.split(/\|/)[4].to_s.length}.max + 4, 7].max, 34].min
|
44
|
+
str = "NAME".ljust(nmax) + "UP DOWN STATE " + "ESSID".ljust(emax) + "QUALITY\r\n"
|
45
|
+
raw.split(/\r?\n/).each { |line|
|
46
|
+
datas = line.split(/\|/)
|
47
|
+
str << datas[0].ljust(nmax) + ('%.1f' % datas[1].to_f).ljust(8) + ('%.1f' % datas[2].to_f).ljust(8) + datas[3].ljust(7)
|
48
|
+
str << datas[4].ljust(emax) + datas[5] if datas[4]
|
49
|
+
str << "\r\n"
|
50
|
+
}
|
51
|
+
str
|
52
|
+
end
|
53
|
+
|
54
|
+
private
|
55
|
+
|
56
|
+
def update
|
57
|
+
file_get_content('/proc/net/dev').gsub(/^.*\|.*?\n/m, '').split("\n").each do |line|
|
58
|
+
datas = line.gsub(/^\s+|\s+$/, '').split(/:\s+|\s+/)
|
59
|
+
name = datas.shift
|
60
|
+
down = datas[0].to_i
|
61
|
+
up = datas[8].to_i
|
62
|
+
|
63
|
+
if not @prev[:up].include? name or not @prev[:down].include? name
|
64
|
+
@prev[:up][name] = @prev[:down][name] = 0
|
65
|
+
end
|
66
|
+
|
67
|
+
@stats[name] = {:up => (up - @prev[:up][name]) / 1024.0,
|
68
|
+
:down => (down - @prev[:down][name]) / 1024.0,
|
69
|
+
:state => get_state(name),
|
70
|
+
:quality => nil,
|
71
|
+
:essid => nil}
|
72
|
+
if @stats[name][:state]
|
73
|
+
@stats[name][:quality] = get_quality(name)
|
74
|
+
@stats[name][:essid] = get_essid(name) if @stats[name][:quality]
|
75
|
+
end
|
76
|
+
@prev[:up][name] = up
|
77
|
+
@prev[:down][name] = down
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
def get_state interface
|
82
|
+
if file_get_content('/proc/net/route') =~ /#{interface}/
|
83
|
+
true
|
84
|
+
else
|
85
|
+
false
|
86
|
+
end
|
87
|
+
end
|
88
|
+
|
89
|
+
def get_quality interface
|
90
|
+
begin
|
91
|
+
file_get_content('/proc/net/wireless').match(/^\s*?#{interface}:.*$/)[0].strip.split(/:\s+|\s+/)[2].gsub('.', '') + "%"
|
92
|
+
rescue
|
93
|
+
nil
|
94
|
+
end
|
95
|
+
end
|
96
|
+
|
97
|
+
def get_essid interface
|
98
|
+
require "socket"
|
99
|
+
|
100
|
+
iwreq = [interface, " " * 32, 32, 0].pack("a16pII")
|
101
|
+
sock = Socket.new(Socket::AF_INET, Socket::SOCK_DGRAM, 0)
|
102
|
+
sock.ioctl(0x8B1B, iwreq)
|
103
|
+
return iwreq.unpack("a16pII")[1].strip
|
104
|
+
end
|
105
|
+
|
106
|
+
def close
|
107
|
+
@serv.kill
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
111
|
+
if __FILE__ == $0
|
112
|
+
x = Net.new
|
113
|
+
10.times {
|
114
|
+
sleep 2
|
115
|
+
puts Net.friendlize Net.protocolize(x.get_infos).gsub(/^net:\s+|END\s+$/m, '')
|
116
|
+
}
|
117
|
+
x.close
|
118
|
+
else
|
119
|
+
Arbi::add_plugin /^net$/i, Net.new
|
120
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
class Ram
|
2
|
+
require 'arbi/common'
|
3
|
+
|
4
|
+
def get_infos
|
5
|
+
mem = file_get_content('/proc/meminfo')
|
6
|
+
total = mem.match(/^MemTotal:\s*?([0-9]+) kB/)[1].to_i
|
7
|
+
free = mem.match(/^MemFree:\s*?([0-9]+) kB/)[1].to_i +
|
8
|
+
mem.match(/^Buffers:\s*?([0-9]+) kB/)[1].to_i +
|
9
|
+
mem.match(/^Cached:\s*?([0-9]+) kB/)[1].to_i
|
10
|
+
perc = 100.0 / total * free
|
11
|
+
{'used' => sprintf('%.1f%%', 100.0 - perc),
|
12
|
+
'free' => sprintf('%.1f%%', perc)}
|
13
|
+
end
|
14
|
+
|
15
|
+
def self.protocolize perc
|
16
|
+
"ram:\r\n#{perc['used']}|#{perc['free']}\r\nEND\r\n"
|
17
|
+
end
|
18
|
+
|
19
|
+
def self.friendlize raw
|
20
|
+
datas = raw.to_s.strip.split(/\r?\n/)[0].split(/\|/)
|
21
|
+
"USED FREE\r\n#{datas[0].ljust(8)}#{datas[1]}"
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
if __FILE__ == $0
|
26
|
+
puts Ram.friendlize Ram.protocolize(Ram.new.get_infos).gsub(/^ram:\s+|END\s+$/m, '')
|
27
|
+
else
|
28
|
+
Arbi::add_plugin /^ram$/i, Ram.new
|
29
|
+
end
|
@@ -0,0 +1,65 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
|
3
|
+
class Thermal
|
4
|
+
require 'arbi/common'
|
5
|
+
|
6
|
+
def initialize
|
7
|
+
@temperatures = Hash.new
|
8
|
+
@tempdirs = Dir.glob "/proc/acpi/thermal_zone/*"
|
9
|
+
@tempdirs.each do |dir|
|
10
|
+
trip_points = file_get_content dir + "/trip_points"
|
11
|
+
name = dir.gsub(/^\/proc\/acpi\/thermal_zone\//, '')
|
12
|
+
@temperatures.merge!({name => {'temperature' => file_get_content(dir + '/temperature').match(
|
13
|
+
/^temperature:\s*?([0-9]{1,3}) C$/)[1],
|
14
|
+
'trip_critical' => trip_points.match(/^critical \(S5\):\s*?([0-9]{1,3}) C$/)[1],}})
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
def get_infos
|
19
|
+
@tempdirs.each do |dir|
|
20
|
+
@temperatures[dir.gsub(/^\/proc\/acpi\/thermal_zone\//, '')]["temperature"] =
|
21
|
+
file_get_content(dir + '/temperature').match(/^temperature:\s*?([0-9]{1,3}) C$/)[1]
|
22
|
+
end
|
23
|
+
@temperatures.merge average
|
24
|
+
end
|
25
|
+
|
26
|
+
def self.protocolize temperatures
|
27
|
+
str = "thermal:\r\n"
|
28
|
+
temperatures.each do |key, value|
|
29
|
+
str += "#{key}|#{value["temperature"]}#{key == "AVERAGE" ? '' : '|' + (value["trip_critical"] || '-')}\r\n"
|
30
|
+
end
|
31
|
+
str += "END\r\n"
|
32
|
+
end
|
33
|
+
|
34
|
+
def self.friendlize raw
|
35
|
+
raw = raw.to_s.strip
|
36
|
+
nmax = raw.lines.map{|x|x.split(/\|/)[0].length}.max + 4
|
37
|
+
tmax = [raw.lines.map{|x|x.split(/\|/)[1].length}.max + 4, 6].max
|
38
|
+
str = "NAME".ljust(nmax) + "TEMP".ljust(tmax) + "CRITICAL\r\n"
|
39
|
+
raw.split(/\r?\n/).each { |line|
|
40
|
+
datas = line.split /\|/
|
41
|
+
str << datas[0].ljust(nmax) + "#{datas[1]}°C".ljust(tmax)
|
42
|
+
str << "#{datas[2]}°C" if datas[2]
|
43
|
+
str << "\r\n"
|
44
|
+
}
|
45
|
+
str
|
46
|
+
end
|
47
|
+
|
48
|
+
private
|
49
|
+
|
50
|
+
def average
|
51
|
+
avg, n = 0, 0
|
52
|
+
@temperatures.each do |value, key|
|
53
|
+
avg += key["temperature"].to_i
|
54
|
+
n += 1
|
55
|
+
end
|
56
|
+
avg = (sprintf '%.1f', (avg.to_f / n)).gsub /\.0$/, ''
|
57
|
+
return ({'AVERAGE' => {'temperature' => avg, 'trip_critical' => false}})
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
if __FILE__ == $0
|
62
|
+
puts Thermal.friendlize Thermal.protocolize(Thermal.new.get_infos).gsub(/^thermal:\s+|END\s+$/m, '')
|
63
|
+
else
|
64
|
+
Arbi::add_plugin /^thermal$/i, Thermal.new
|
65
|
+
end
|
metadata
ADDED
@@ -0,0 +1,104 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: arbi
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
hash: 19
|
5
|
+
prerelease: false
|
6
|
+
segments:
|
7
|
+
- 1
|
8
|
+
- 0
|
9
|
+
- 2
|
10
|
+
version: 1.0.2
|
11
|
+
platform: ruby
|
12
|
+
authors:
|
13
|
+
- shura
|
14
|
+
autorequire:
|
15
|
+
bindir: bin
|
16
|
+
cert_chain: []
|
17
|
+
|
18
|
+
date: 2010-07-10 00:00:00 +02:00
|
19
|
+
default_executable:
|
20
|
+
dependencies:
|
21
|
+
- !ruby/object:Gem::Dependency
|
22
|
+
name: getopt
|
23
|
+
prerelease: false
|
24
|
+
requirement: &id001 !ruby/object:Gem::Requirement
|
25
|
+
none: false
|
26
|
+
requirements:
|
27
|
+
- - ">="
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
hash: 3
|
30
|
+
segments:
|
31
|
+
- 0
|
32
|
+
version: "0"
|
33
|
+
type: :runtime
|
34
|
+
version_requirements: *id001
|
35
|
+
- !ruby/object:Gem::Dependency
|
36
|
+
name: sys-filesystem
|
37
|
+
prerelease: false
|
38
|
+
requirement: &id002 !ruby/object:Gem::Requirement
|
39
|
+
none: false
|
40
|
+
requirements:
|
41
|
+
- - ">="
|
42
|
+
- !ruby/object:Gem::Version
|
43
|
+
hash: 3
|
44
|
+
segments:
|
45
|
+
- 0
|
46
|
+
version: "0"
|
47
|
+
type: :runtime
|
48
|
+
version_requirements: *id002
|
49
|
+
description: ""
|
50
|
+
email: shura1991@gmail.com
|
51
|
+
executables:
|
52
|
+
- arbid
|
53
|
+
- arbi
|
54
|
+
extensions: []
|
55
|
+
|
56
|
+
extra_rdoc_files: []
|
57
|
+
|
58
|
+
files:
|
59
|
+
- lib/arbi.rb
|
60
|
+
- lib/arbi/common.rb
|
61
|
+
- lib/arbi/plugins/ram.rb
|
62
|
+
- lib/arbi/plugins/diskspace.rb
|
63
|
+
- lib/arbi/plugins/cpu.rb
|
64
|
+
- lib/arbi/plugins/batteries.rb
|
65
|
+
- lib/arbi/plugins/net.rb
|
66
|
+
- lib/arbi/plugins/thermal.rb
|
67
|
+
- bin/arbid
|
68
|
+
- bin/arbi
|
69
|
+
has_rdoc: true
|
70
|
+
homepage: http://github.com/shurizzle/arbi
|
71
|
+
licenses: []
|
72
|
+
|
73
|
+
post_install_message:
|
74
|
+
rdoc_options: []
|
75
|
+
|
76
|
+
require_paths:
|
77
|
+
- lib
|
78
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
79
|
+
none: false
|
80
|
+
requirements:
|
81
|
+
- - ">="
|
82
|
+
- !ruby/object:Gem::Version
|
83
|
+
hash: 3
|
84
|
+
segments:
|
85
|
+
- 0
|
86
|
+
version: "0"
|
87
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
88
|
+
none: false
|
89
|
+
requirements:
|
90
|
+
- - ">="
|
91
|
+
- !ruby/object:Gem::Version
|
92
|
+
hash: 3
|
93
|
+
segments:
|
94
|
+
- 0
|
95
|
+
version: "0"
|
96
|
+
requirements: []
|
97
|
+
|
98
|
+
rubyforge_project:
|
99
|
+
rubygems_version: 1.3.7
|
100
|
+
signing_key:
|
101
|
+
specification_version: 3
|
102
|
+
summary: ""
|
103
|
+
test_files: []
|
104
|
+
|