APIを介してReviewBoardに差分を投稿するのに苦労しています。サーバーにログインして新しい投稿を作成できましたが、diffファイルの内容を正しく投稿できませんでした。
この種のアプリケーションを作成するのは初めてですが、私の目標は、次のようなワンステップのスクリプトを作成することです。
- svnリポジトリとファイルを比較(事前コミット)し、
- ReviewBoardにレビューリクエストを追加し、現在のファイルとの差分を投稿します。
後で、スクリプトをsvnpre-commitフックの一部にすることができます。
私のPythonの試みは次のようになります:
import urllib.request
import urllib.parse
import os.path
... login to the reviewboard server with
urllib.request.HTTPBasicAuthHandler ...
diff_path = '/path/to/file'
diff_name = 'my.diff'
diff_path = os.path.join(diff_path, diff_name)
diff_val = open(diff_path,'r')
# load the diff into the http data POST request
diff_header = \
'-- SoMe BoUnDaRy \n' \
+ 'Content-Disposition: form-data; name=path; filename=' \
+ '"' + diff_name + '"\n\n' \
+ diff_val.read() + '\n' \
+ '-- SoMe BoUnDaRy --'
data ={'path': diff_header, 'basedir': '/path/to/file/in/rep'}
print( data['path'] )
data = urllib.parse.urlencode(data)
data = data.encode('utf-8')
opener.open( \
'http://xxx.xxx.x.xxx/api/review-requests/26/diffs/', data)
このコードを使用すると、BAD REQUEST(400)エラーが発生します。具体的には、「1つ以上のフィールドにエラーがありました」(105)です。
ReviewBoardAPIと通信できるライブラリがいくつかあることを私は知っています。事後レビューが存在することも承知しています。他の開発者に別のPythonライブラリを配布する必要はなく、複数の場所からファイルを比較する場合、レビュー後の柔軟性が低くなるようです。
以下の提案から、ここにサーバーの応答を追加しました。
CREATING PASSWD MANAGER...
CREATING PASSWD MANAGER... done
CREATING PASSWD HANDLER...
CREATING PASSWD HANDLER... done
CREATING URL OPENER...
CREATING URL OPENER... done
LOADING DIFF...
send: b'POST /api/review-requests/26/diffs/ HTTP/1.1\r\nAccept-Encoding:
identity\r\nContent-Length: 723\r\nHost: xxx.xxx.x.xxx\r\nContent-Type:
application/x-www-form-urlencoded\r\nConnection: close\r\nUser-Agent:
[empty no username+password] Python-urllib/3.2\r\n\r\
npath=--+SoMe+BoUnDaRy+++%...[the rest of my post]
reply: 'HTTP/1.1 401 UNAUTHORIZED\r\n'
header: Date header: Server header: Content-Language header: Expires header:
Vary header: Cache-Control header: WWW-Authenticate header:
Content-Length header: Last-Modified header: Connection header:
Content-Type send: b'POST /api/review-requests/26/diffs/
HTTP/1.1\r\nAccept-Encoding: identity\r\nContent-Length: 723\r\nHost:
xxx.xxx.x.xxx\r\nUser-Agent: Python-urllib/3.2\r\nConnection:
close\r\nContent-Type: application/x-www-form-urlencoded\r\nAuthorization:
Basic [with username+password]\r\n\r\npath=
--+SoMe+BoUnDaRy+++%0AContent-Disposition%...
reply: 'HTTP/1.1 400 BAD REQUEST\r\n'
header: Date header: Server header: Content-Language header: Expires header:
Vary header: Cache-Control header: Set-Cookie header: Content-Length header:
Last-Modified header: Connection header: Content-Type HTTPError thrown
一見すると、パスワードハンドラーに何かが起こっていると思います。何が起こっているのかわかりません。念のため、これが私の認証を生成する方法です。
manager_passwd = urllib.request.HTTPPasswordMgr()
manager_passwd.add_password(...)
handler_passwd = urllib.request.HTTPBasicAuthHandler(manager_passwd)
opener = urllib.request.build_opener(handler_passwd)
認証は機能しているようです。新しいレビュー投稿を作成してテストしました。したがって、認証が失敗するのは、diffを投稿するときです。