以前の記事、shellコマンドをJSON-RPC経由で実行する
にて作ったshell-jsonrpc
ですが、ふとバイナリファイルの扱いがどうなっているか不安になって調べてみました→だめでした、ので改良してみます。
$ curl http://0.0.0.0:8999 -d '{"jsonrpc":"2.0", "id": "abc", "method":"shell", "params": [{"command": "cat a.png"}] }'|jq .
例外が発生します
1
2
3
4
5
6
7
8
9
10
11
| [2021-01-28 00:16:54] ERROR JSON::GeneratorError: source sequence is illegal/malformed utf-8
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/multi_json-1.15.0/lib/multi_json/adapters/json_common.rb:19:in `to_json'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/multi_json-1.15.0/lib/multi_json/adapters/json_common.rb:19:in `dump'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/multi_json-1.15.0/lib/multi_json/adapter.rb:25:in `dump'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/multi_json-1.15.0/lib/multi_json.rb:139:in `dump'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/jimson-0.11.0/lib/jimson/server.rb:112:in `process'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/jimson-0.11.0/lib/jimson/server.rb:87:in `call'
/Users/tumf/.rvm/gems/ruby-2.7.2/gems/rack-1.6.13/lib/rack/handler/webrick.rb:88:in `service'
/Users/tumf/.rvm/rubies/ruby-2.7.2/lib/ruby/2.7.0/webrick/httpserver.rb:140:in `service'
/Users/tumf/.rvm/rubies/ruby-2.7.2/lib/ruby/2.7.0/webrick/httpserver.rb:96:in `run'
/Users/tumf/.rvm/rubies/ruby-2.7.2/lib/ruby/2.7.0/webrick/server.rb:307:in `block in start_thread'
|
binding.pry
で覗いてみると標準出力がバイナリになっていますね。
そこでバイナリを判別して、Base64エンコードするようにしてみます。バイナリ判定アルゴリズムはptools
を参考にしました。
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
| require 'bundler/setup'
require 'jimson'
require 'json'
require "open3"
require 'base64'
# https://github.com/djberg96/ptools/blob/master/lib/ptools.rb#L77
def binary?(s, percentage = 0.30)
s = s.encode('US-ASCII', :undef => :replace).split(//)
((s.size - s.grep(" ".."~").size) / s.size.to_f) > percentage
rescue Encoding::InvalidByteSequenceError =>e
true
end
class MyHandler
extend Jimson::Handler
def shell *params
params.collect { |param|
c = param["command"]
i = if param["stdin_binary"]
Base64.decode64(param["stdin"])
else
param["stdin"]
end
o, e, s = Open3.capture3(c, stdin_data: i)
if binary?(o)
o = Base64.encode64(o)
else
json = JSON.parse(o) rescue nil
end
{ command: c, "stdout.json": json, stdout: o, stderr: e, status: { exitstatus: s.to_i, pid: s.pid } }
}
rescue =>e
p e
p e.backtrace
raise
end
end
server = Jimson::Server.new(MyHandler.new)
server.start # serve with webrick on http://0.0.0.0:8999/
|
しかしこのバイナリー判定30%以上の非ASCIIがあるかどうかでバイナリと判定するっていうの面白いですね・・・
# https://github.com/djberg96/ptools/blob/master/lib/ptools.rb#L77
def binary?(s, percentage = 0.30)
s = s.encode('US-ASCII', :undef => :replace).split(//)
((s.size - s.grep(" ".."~").size) / s.size.to_f) > percentage
rescue Encoding::InvalidByteSequenceError =>e
true
end
さて、これで冒頭のテストをします
$ curl http://0.0.0.0:8999 -d '{"jsonrpc":"2.0", "id": "abc", "method":"shell", "params": [{"command": "cat a.png"}] }'|jq .
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
| {
"jsonrpc": "2.0",
"result": [
{
"command": "cat",
"stdout.json": null,
"stdout": "iVBORw0KGgoAAAANSUhEUgAABgQAAALmCAIAAAAC...UeAHs3Qd8\n",
"stderr": "",
"status": {
"exitstatus": 0,
"pid": 53175
}
}
],
"id": "abc"
}
|
うまくいきました👍