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
|
# -*- coding: utf-8 -*-
require 'net/http'
Plugin.create :bugreport do
@bugreport_uri = Diva::URI('https://mikutter.hachune.net/')
Delayer.new do |service|
popup if crashed_exception.is_a? Exception
rescue => e
# バグ報告中にバグで死んだらつらいもんな
error e
end
def popup
alert_thread = if(Thread.main != Thread.current) then Thread.current end
dialog = Gtk::Dialog.new(title: "bug report")
dialog.set_size_request(600, 400)
dialog.window_position = Gtk::WindowPosition::CENTER
dialog.child.pack_start(main, expand: true, fill: true, padding: 30)
dialog.add_button(Gtk::Stock::OK, Gtk::ResponseType::OK)
dialog.add_button(Gtk::Stock::CANCEL, Gtk::ResponseType::CANCEL)
dialog.default_response = Gtk::ResponseType::OK
quit = lambda{
dialog.hide.destroy
Gtk.main_iteration_do(false)
if alert_thread
alert_thread.run
else
Gtk.main_quit
end }
dialog.signal_connect("response"){ |widget, response|
if response == Gtk::ResponseType::OK
send
else
File.delete(File.expand_path(File.join(Environment::TMPDIR, 'mikutter_error'))) rescue nil
File.delete(File.expand_path(File.join(Environment::TMPDIR, 'crashed_exception'))) rescue nil end
quit.call }
dialog.signal_connect("destroy") {
false
}
dialog.show_all
if(alert_thread)
Thread.stop
else
Gtk::main
end
end
def imsorry
_("%{mikutter} が突然終了してしまったみたいで ヽ('ω')ノ三ヽ('ω')ノもうしわけねぇもうしわけねぇ")%{mikutter: Environment::NAME}+"\n"+
_('OKボタンを押したら、自動的に以下のテキストが送られます。これがバグを直すのにとっても役に立つんですよ。よかったら送ってくれません?')
end
def main
sw = Gtk::ScrolledWindow.new
sw.set_policy :never, :always
sw.vexpand = true
sw.valign = :fill
sw << Gtk::IntelligentTextview.new(backtrace)
sorry = Gtk::IntelligentTextview.new(imsorry)
sorry.hexpand = true
grid = Gtk::Grid.new
grid.orientation = :vertical
grid << sorry << sw
end
def send
Thread.new do
exception = crashed_exception
m = exception.backtrace.first.match(/(.+?):(\d+)/)
crashed_file, crashed_line = m[1], m[2]
param = {
'backtrace' => JSON.generate(exception.backtrace.map{ |msg| msg.gsub(FOLLOW_DIR, '{MIKUTTER_DIR}') }),
'file' => crashed_file.gsub(FOLLOW_DIR, '{MIKUTTER_DIR}'),
'line' => crashed_line,
'exception_class' => exception.class,
'description' => exception.to_s,
'ruby_version' => RUBY_VERSION,
'rubygtk_version' => GLib::BINDING_VERSION.join('.'),
'platform' => RUBY_PLATFORM,
'url' => 'exception',
'version' => Environment::VERSION
}
case exception
when TypeStrictError
param['causing_value'] = exception.value
end
http = Net::HTTP.new(@bugreport_uri.host, @bugreport_uri.port)
http.use_ssl = @bugreport_uri.scheme == 'https'
req = Net::HTTP::Post.new(@bugreport_uri.path)
req.set_form_data(param)
res = http.request(req)
File.delete(File.expand_path(File.join(Environment::TMPDIR, 'mikutter_error'))) rescue nil
File.delete(File.expand_path(File.join(Environment::TMPDIR, 'crashed_exception'))) rescue nil
Plugin.activity :system, _("エラー報告を送信しました。ありがとう♡")
Plugin.call :send_bugreport, param
rescue Timeout::Error, StandardError => e
Plugin.activity :system, _("ピャアアアアアアアアアアアアアアアアアアアアアアアwwwwwwwwwwwwwwwwwwwwww")
Plugin.activity :error, e.to_s, exception: e
end
end
def backtrace
"#{crashed_exception.class} #{crashed_exception.to_s}\n" +
crashed_exception.backtrace.map{ |msg| msg.gsub(FOLLOW_DIR, '{MIKUTTER_DIR}') }.join("\n")
end
def crashed_exception
@crashed_exception ||= object_get_contents(File.expand_path(File.join(Environment::TMPDIR, 'crashed_exception'))) rescue nil
end
end
|