sample.lua 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. --[[
  2. willSendRequestToBbsCgi 関数
  3. 5ch/bbspinkのbbs.cgiにPOSTリクエストが送られる直前にproxy2chから呼び出される
  4. 関数です。引数requestと同じ構造のtable型の変数を返すことで、proxy2chがbbs.cgi
  5. に対して送るリクエストヘッダ/ボディを自由に改変 (ヘッダの追加/削除含む) する
  6. ことができます。--bbscgi-headerオプションとは異なり、板ごとに動作を変更する
  7. ことも可能です。スクリプトは毎回読み込まれるため、proxy2chを起動し直さなくて
  8. も変更は反映されます。
  9. * 引数 request
  10. 次のような構造を持つtable型の変数です。送信される予定のリクエストヘッダ/
  11. ボディが格納されています。
  12. {
  13. "body" = "リクエストボディ",
  14. "headers" = {
  15. "リクエストヘッダ1" = "値1",
  16. "リクエストヘッダ2" = "値2",
  17. :
  18. }
  19. }
  20. この引数をそのまま返せば「何もしない」関数になります。
  21. 特定のヘッダを送信したくない場合は値をnilに設定して削除するか、あるいは空の
  22. 文字列""を代入します。
  23. * 引数 host
  24. リクエスト先のホスト名が格納されているstring型の変数です。
  25. * 引数 board
  26. リクエストボディのbbsフィールドの値に相当するstring型の変数です。
  27. * 引数 thread
  28. リクエストボディのkeyフィールドの値に相当するstring型の変数です。
  29. * 戻り値
  30. 引数requestと同じ構造を持つtable型の変数を返してください。requestそのもの
  31. を改変してもいいですし、新しいtableを1から作ってもいいです。
  32. リクエストボディを改変する場合、application/x-www-form-urlencodedの形式を
  33. 満たしている必要があります。
  34. このスクリプトの中からはproxy2chというグローバル変数を参照可能であり、いくつかの
  35. ユーティリティ関数や変数が proxy2ch.(関数名/変数名) の形でアクセスできます。
  36. proxy2ch.hmacSHA256(key, message) 関数
  37. HMAC-SHA256に基づいてメッセージダイジェストを算出します。
  38. * 引数 key
  39. HMAC算出用の鍵をstring型で渡します。
  40. * 引数 message
  41. ダイジェストを算出したいメッセージをstring型で渡します。
  42. * 戻り値
  43. 算出したメッセージダイジェストの値をstring型で返します。
  44. hex表記したもの、つまり0-9,a-fで構成される長さ64の文字列になります。
  45. proxy2ch.decodeURIComponent(input [, decodePlus]) 関数
  46. URLエンコードされた文字列をデコードします。
  47. * 引数 input
  48. デコードしたい文字列をstring型で渡します。
  49. * 引数 decodePlus (省略可能)
  50. input中に現れる"+"を" "に置き換えるかどうかをboolean型で渡します。
  51. application/x-www-form-urlencoded形式を想定しているため、この引数を省略した
  52. 場合はtrueが渡されたものとして動作します。
  53. * 戻り値
  54. デコード済み文字列をstring型で返します。
  55. proxy2ch.encodeURIComponent(input [, spaceAsPlus]) 関数
  56. 文字列をURLエンコードします。
  57. * 引数 input
  58. エンコードしたい文字列をstring型で渡します。
  59. * 引数 spaceAsPlus (省略可能)
  60. input中に現れる" "を"%20"ではなく"+"に置き換えるかどうかをboolean型で渡しま
  61. す。application/x-www-form-urlencoded形式を想定しているため、この引数を省略
  62. した場合はtrueが渡されたものとして動作します。
  63. * 戻り値
  64. エンコード済み文字列をstring型で返します。
  65. proxy2ch.convertShiftJISToUTF8(input) 関数
  66. Shift JISでエンコードされた文字列をUTF-8に変換します。
  67. * 引数 input
  68. UTF-8に変換したい文字列をstring型で渡します。
  69. * 戻り値
  70. UTF-8に変換された文字列をstring型で返します。
  71. 変換に失敗した場合はnilを返します。
  72. proxy2ch.isKeyExpired(key) 関数
  73. 指定したMonaKeyが有効期限切れ扱いされたことがあるかどうかをチェックします。
  74. 通常は使う必要はないですが、MonaKeyの管理をproxy2ch.monaKeyに丸投げせずに
  75. スクリプト側で行いたい場合に役に立つ可能性があります。
  76. * 引数 key
  77. チェックしたいMonaKeyをstring型で渡します。
  78. * 戻り値
  79. 有効期限切れ扱いされたことがあるかどうかをboolean型で返します。
  80. proxy2ch.monaKey string型変数
  81. 書き込み時に払い出されたMonaKey(仮称)の値を格納している変数です。
  82. 払い出し前の初期値は便宜上 "00000000-0000-0000-0000-000000000000" です。
  83. ]]
  84. function createTableFromBody(str)
  85. local t = {}
  86. for s in string.gmatch(str, "([^&]+)") do
  87. local idx = string.find(s, "=")
  88. local name = string.sub(s, 1, idx-1)
  89. local value = string.sub(s, idx+1)
  90. t[name] = value
  91. end
  92. return t
  93. end
  94. function createBodyFromTableWithOrder(tBody, order)
  95. order = order or {}
  96. local ret = ""
  97. for i, name in ipairs(order) do
  98. if tBody[name] ~= nil then
  99. if ret ~= "" then
  100. ret = ret .. "&"
  101. end
  102. ret = ret .. name .. "=" .. tBody[name]
  103. tBody[name] = nil
  104. end
  105. end
  106. for name, value in pairs(tBody) do
  107. if ret ~= "" then
  108. ret = ret .. "&"
  109. end
  110. ret = ret .. name .. "=" .. value
  111. end
  112. return ret
  113. end
  114. function generatePostSignatureFromTable(tBody, userAgent, hmacKey)
  115. local bbs = tBody["bbs"] or ""
  116. local key = tBody["key"] or ""
  117. local time = tBody["time"] or ""
  118. local from = tBody["FROM"] or ""
  119. local mail = tBody["mail"] or ""
  120. local message = tBody["MESSAGE"] or ""
  121. local subject = tBody["subject"] or ""
  122. local userAgent = userAgent or ""
  123. local monaKey = proxy2ch.monaKey
  124. local unused = ""
  125. local nonce = string.format("%d.%03d", os.time(), math.random(0, 999))
  126. local data = table.concat({bbs, key, time, from, mail, message, subject, userAgent, monaKey, unused, nonce}, "<>")
  127. return proxy2ch.hmacSHA256(hmacKey, data), nonce
  128. end
  129. function willSendRequestToBbsCgi(request, host, board, thread)
  130. --リクエストヘッダのUser-Agentの値を変更する
  131. request.headers["User-Agent"] = "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:88.0) Gecko/20100101 Firefox/88.0"
  132. --リクエストヘッダのRefererの値を変更する
  133. request.headers["Referer"] = "https://" .. host .. "/test/read.cgi/" .. board .. "/" .. thread .. "/l50"
  134. --リクエストボディの末尾に"&oekaki_thread1="を追加する
  135. request.body = request.body .. "&oekaki_thread1="
  136. --リクエストボディをtableに変換する
  137. --"name1=value1&name=value2"形式のstringが{"name1" = "value1", "name2" = "value2"}形式のtableに変換される
  138. local tBody = createTableFromBody(request.body)
  139. --リクエストボディの各フィールドのURLエンコードを解除し、UTF-8に変換する
  140. for name, value in pairs(tBody) do
  141. tBody[name] = proxy2ch.convertShiftJISToUTF8(proxy2ch.decodeURIComponent(value))
  142. end
  143. --リクエストボディとUser-Agentからリクエストのsignatureを生成する
  144. --リクエストボディはUTF-8である必要はないがURLエンコードは解除しておく必要がある
  145. local postSig, postNonce = generatePostSignatureFromTable(tBody, request.headers["User-Agent"], "this is an HMAC key")
  146. --リクエストボディの各フィールドを再度URLエンコードする
  147. for name, value in pairs(tBody) do
  148. tBody[name] = proxy2ch.encodeURIComponent(value)
  149. end
  150. --tableからフィールドの順序を指定して新しいリクエストボディを生成する
  151. request.body = createBodyFromTableWithOrder(tBody, {"FROM", "mail", "MESSAGE", "bbs", "key", "time", "submit"})
  152. --Content-TypeヘッダのcharsetにUTF-8を指定しておく
  153. request.headers["Content-Type"] = "application/x-www-form-urlencoded; charset=UTF-8"
  154. --hostがbbspink.comでなければ新しい書き込み仕様に必要なヘッダを設定
  155. if host:sub(-#".bbspink.com") ~= ".bbspink.com" then
  156. request.headers["X-PostSig"] = postSig
  157. request.headers["X-APIKey"] = "this is an API key"
  158. request.headers["X-PostNonce"] = postNonce
  159. request.headers["X-MonaKey"] = proxy2ch.monaKey
  160. end
  161. return request
  162. end