pw 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527
  1. #!/usr/bin/env python
  2. #
  3. # Patchwork command line client
  4. # Copyright (C) 2008 Nate Case <ncase@xes-inc.com>
  5. #
  6. # This file is part of the Patchwork package.
  7. #
  8. # Patchwork is free software; you can redistribute it and/or modify
  9. # it under the terms of the GNU General Public License as published by
  10. # the Free Software Foundation; either version 2 of the License, or
  11. # (at your option) any later version.
  12. #
  13. # Patchwork is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU General Public License
  19. # along with Patchwork; if not, write to the Free Software
  20. # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  21. import os
  22. import sys
  23. import xmlrpclib
  24. import getopt
  25. import string
  26. import tempfile
  27. import subprocess
  28. import base64
  29. import ConfigParser
  30. import datetime
  31. import re
  32. # Default Patchwork remote XML-RPC server URL
  33. # This script will check the PW_XMLRPC_URL environment variable
  34. # for the URL to access. If that is unspecified, it will fallback to
  35. # the hardcoded default value specified here.
  36. DEFAULT_URL = "http://patchwork/xmlrpc/"
  37. CONFIG_FILES = [os.path.expanduser('~/.pwclientrc')]
  38. class Filter:
  39. """Filter for selecting patches."""
  40. def __init__(self):
  41. # These fields refer to specific objects, so they are special
  42. # because we have to resolve them to IDs before passing the
  43. # filter to the server
  44. self.state = ""
  45. self.project = ""
  46. # The dictionary that gets passed to via XML-RPC
  47. self.d = {}
  48. def add(self, field, value):
  49. if field == 'state':
  50. self.state = value
  51. elif field == 'project':
  52. self.project = value
  53. else:
  54. # OK to add directly
  55. self.d[field] = value
  56. def resolve_ids(self, rpc):
  57. """Resolve State, Project, and Person IDs based on filter strings."""
  58. if self.state != "":
  59. id = state_id_by_name(rpc, self.state)
  60. if id == 0:
  61. sys.stderr.write("Note: No State found matching %s*, " \
  62. "ignoring filter\n" % self.state)
  63. else:
  64. self.d['state_id'] = id
  65. if self.project != "":
  66. id = project_id_by_name(rpc, self.project)
  67. if id == 0:
  68. sys.stderr.write("Note: No Project found matching %s, " \
  69. "ignoring filter\n" % self.project)
  70. else:
  71. self.d['project_id'] = id
  72. def __str__(self):
  73. """Return human-readable description of the filter."""
  74. return str(self.d)
  75. class BasicHTTPAuthTransport(xmlrpclib.SafeTransport):
  76. def __init__(self, username = None, password = None, use_https = False):
  77. self.username = username
  78. self.password = password
  79. self.use_https = use_https
  80. xmlrpclib.SafeTransport.__init__(self)
  81. def authenticated(self):
  82. return self.username != None and self.password != None
  83. def send_host(self, connection, host):
  84. xmlrpclib.Transport.send_host(self, connection, host)
  85. if not self.authenticated():
  86. return
  87. credentials = '%s:%s' % (self.username, self.password)
  88. auth = 'Basic ' + base64.encodestring(credentials).strip()
  89. connection.putheader('Authorization', auth)
  90. def make_connection(self, host):
  91. if self.use_https:
  92. fn = xmlrpclib.SafeTransport.make_connection
  93. else:
  94. fn = xmlrpclib.Transport.make_connection
  95. return fn(self, host)
  96. def usage():
  97. sys.stderr.write("Usage: %s <action> [options]\n\n" % \
  98. (os.path.basename(sys.argv[0])))
  99. sys.stderr.write("Where <action> is one of:\n")
  100. sys.stderr.write(
  101. """ apply <ID> : Apply a patch (in the current dir, using -p1)
  102. get <ID> : Download a patch and save it locally
  103. projects : List all projects
  104. states : Show list of potential patch states
  105. list [str] : List patches, using the optional filters specified
  106. below and an optional substring to search for patches
  107. by name
  108. search [str] : Same as 'list'
  109. view <ID> : View a patch
  110. update [-s state] [-c commit-ref] <ID>
  111. : Update patch\n""")
  112. sys.stderr.write("""\nFilter options for 'list' and 'search':
  113. -s <state> : Filter by patch state (e.g., 'New', 'Accepted', etc.)
  114. -p <project> : Filter by project name (see 'projects' for list)
  115. -w <who> : Filter by submitter (name, e-mail substring search)
  116. -d <who> : Filter by delegate (name, e-mail substring search)
  117. -n <max #> : Restrict number of results\n""")
  118. sys.stderr.write("""\nActions that take an ID argument can also be \
  119. invoked with:
  120. -h <hash> : Lookup by patch hash\n""")
  121. sys.exit(1)
  122. def project_id_by_name(rpc, linkname):
  123. """Given a project short name, look up the Project ID."""
  124. if len(linkname) == 0:
  125. return 0
  126. projects = rpc.project_list(linkname, 0)
  127. for project in projects:
  128. if project['linkname'] == linkname:
  129. return project['id']
  130. return 0
  131. def state_id_by_name(rpc, name):
  132. """Given a partial state name, look up the state ID."""
  133. if len(name) == 0:
  134. return 0
  135. states = rpc.state_list(name, 0)
  136. for state in states:
  137. if state['name'].lower().startswith(name.lower()):
  138. return state['id']
  139. return 0
  140. def person_ids_by_name(rpc, name):
  141. """Given a partial name or email address, return a list of the
  142. person IDs that match."""
  143. if len(name) == 0:
  144. return []
  145. people = rpc.person_list(name, 0)
  146. return map(lambda x: x['id'], people)
  147. def list_patches(patches):
  148. """Dump a list of patches to stdout."""
  149. print("%-5s %-12s %s" % ("ID", "State", "Name"))
  150. print("%-5s %-12s %s" % ("--", "-----", "----"))
  151. for patch in patches:
  152. print("%-5d %-12s %s" % (patch['id'], patch['state'], patch['name']))
  153. def action_list(rpc, filter, submitter_str, delegate_str):
  154. filter.resolve_ids(rpc)
  155. if submitter_str != "":
  156. ids = person_ids_by_name(rpc, submitter_str)
  157. if len(ids) == 0:
  158. sys.stderr.write("Note: Nobody found matching *%s*\n", \
  159. submitter_str)
  160. else:
  161. for id in ids:
  162. person = rpc.person_get(id)
  163. print "Patches submitted by %s <%s>:" % \
  164. (person['name'], person['email'])
  165. f = filter
  166. f.add("submitter_id", id)
  167. patches = rpc.patch_list(f.d)
  168. list_patches(patches)
  169. return
  170. if delegate_str != "":
  171. ids = person_ids_by_name(rpc, delegate_str)
  172. if len(ids) == 0:
  173. sys.stderr.write("Note: Nobody found matching *%s*\n", \
  174. delegate_str)
  175. else:
  176. for id in ids:
  177. person = rpc.person_get(id)
  178. print "Patches delegated to %s <%s>:" % \
  179. (person['name'], person['email'])
  180. f = filter
  181. f.add("delegate_id", id)
  182. patches = rpc.patch_list(f.d)
  183. list_patches(patches)
  184. return
  185. patches = rpc.patch_list(filter.d)
  186. list_patches(patches)
  187. def action_projects(rpc):
  188. projects = rpc.project_list("", 0)
  189. print("%-5s %-24s %s" % ("ID", "Name", "Description"))
  190. print("%-5s %-24s %s" % ("--", "----", "-----------"))
  191. for project in projects:
  192. print("%-5d %-24s %s" % (project['id'], \
  193. project['linkname'], \
  194. project['name']))
  195. def action_states(rpc):
  196. states = rpc.state_list("", 0)
  197. print("%-5s %s" % ("ID", "Name"))
  198. print("%-5s %s" % ("--", "----"))
  199. for state in states:
  200. print("%-5d %s" % (state['id'], state['name']))
  201. def action_get(rpc, patch_id):
  202. patch = rpc.patch_get(patch_id)
  203. s = rpc.patch_get_mbox(patch_id)
  204. if patch == {} or len(s) == 0:
  205. sys.stderr.write("Unable to get patch %d\n" % patch_id)
  206. sys.exit(1)
  207. base_fname = fname = os.path.basename(patch['filename'])
  208. i = 0
  209. while os.path.exists(fname):
  210. fname = "%s.%d" % (base_fname, i)
  211. i += 1
  212. try:
  213. f = open(fname, "w")
  214. except:
  215. sys.stderr.write("Unable to open %s for writing\n" % fname)
  216. sys.exit(1)
  217. try:
  218. f.write(unicode(s).encode("utf-8"))
  219. f.close()
  220. print "Saved patch to %s" % fname
  221. except:
  222. sys.stderr.write("Failed to write to %s\n" % fname)
  223. sys.exit(1)
  224. def action_apply(rpc, patch_id):
  225. patch = rpc.patch_get(patch_id)
  226. if patch == {}:
  227. sys.stderr.write("Error getting information on patch ID %d\n" % \
  228. patch_id)
  229. sys.exit(1)
  230. print "Applying patch #%d to current directory" % patch_id
  231. print "Description: %s" % patch['name']
  232. s = rpc.patch_get_mbox(patch_id)
  233. if len(s) > 0:
  234. proc = subprocess.Popen(['patch', '-p1'], stdin = subprocess.PIPE)
  235. proc.communicate(s)
  236. else:
  237. sys.stderr.write("Error: No patch content found\n")
  238. sys.exit(1)
  239. def action_update_patch(rpc, patch_id, state = None, commit = None):
  240. patch = rpc.patch_get(patch_id)
  241. if patch == {}:
  242. sys.stderr.write("Error getting information on patch ID %d\n" % \
  243. patch_id)
  244. sys.exit(1)
  245. params = {}
  246. if state:
  247. state_id = state_id_by_name(rpc, state)
  248. if state_id == 0:
  249. sys.stderr.write("Error: No State found matching %s*\n" % state)
  250. sys.exit(1)
  251. params['state'] = state_id
  252. if commit:
  253. params['commit_ref'] = commit
  254. success = False
  255. try:
  256. success = rpc.patch_set(patch_id, params)
  257. except xmlrpclib.Fault, f:
  258. sys.stderr.write("Error updating patch: %s\n" % f.faultString)
  259. if not success:
  260. sys.stderr.write("Patch not updated\n")
  261. def patch_id_from_hash(rpc, project, hash):
  262. try:
  263. patch = rpc.patch_get_by_project_hash(project, hash)
  264. except xmlrpclib.Fault:
  265. # the server may not have the newer patch_get_by_project_hash function,
  266. # so fall back to hash-only.
  267. patch = rpc.patch_get_by_hash(hash)
  268. if patch == {}:
  269. return None
  270. return patch['id']
  271. def branch_with(patch_id, rpc):
  272. s = rpc.patch_get_mbox(patch_id)
  273. if len(s) > 0:
  274. print unicode(s).encode("utf-8")
  275. patch = rpc.patch_get(patch_id)
  276. # Find the latest commit from the day before the patch
  277. proc = subprocess.Popen(['git', 'log', '--until=' + patch['date'],
  278. '-1', '--format=%H', 'master'],
  279. stdout = subprocess.PIPE)
  280. sha = proc.stdout.read()[:-1]
  281. # Create a topic branch named after this commit
  282. proc = subprocess.Popen(['git', 'checkout', '-b', 't/patch%s' %
  283. patch_id, sha])
  284. sts = os.waitpid(proc.pid, 0)
  285. if sts[1] != 0:
  286. sys.stderr.write("Could not create branch for patch\n")
  287. return
  288. # Apply the patch to the branch
  289. fname = '/tmp/patch'
  290. try:
  291. f = open(fname, "w")
  292. except:
  293. sys.stderr.write("Unable to open %s for writing\n" % fname)
  294. sys.exit(1)
  295. try:
  296. f.write(unicode(s).encode("utf-8"))
  297. f.close()
  298. print "Saved patch to %s" % fname
  299. except:
  300. sys.stderr.write("Failed to write to %s\n" % fname)
  301. sys.exit(1)
  302. proc = subprocess.Popen(['git', 'am', '/tmp/patch'])
  303. sts = os.waitpid(proc.pid, 0)
  304. if sts[1] != 0:
  305. sys.stderr.write("Failed to apply patch to branch\n")
  306. proc = subprocess.Popen(['git', 'checkout', 'master'])
  307. os.waitpid(proc.pid, 0)
  308. proc = subprocess.Popen(['git', 'branch', '-D', 't/patch%s' %
  309. patch_id, sha])
  310. os.waitpid(proc.pid, 0)
  311. return
  312. proc = subprocess.Popen(['git', 'rebase', 'master'])
  313. sts = os.waitpid(proc.pid, 0)
  314. print sha
  315. auth_actions = ['update']
  316. def main():
  317. try:
  318. opts, args = getopt.getopt(sys.argv[2:], 's:p:w:d:n:c:h:')
  319. except getopt.GetoptError, err:
  320. print str(err)
  321. usage()
  322. if len(sys.argv) < 2:
  323. usage()
  324. action = sys.argv[1].lower()
  325. # set defaults
  326. filt = Filter()
  327. submitter_str = ""
  328. delegate_str = ""
  329. project_str = ""
  330. commit_str = ""
  331. state_str = "New"
  332. hash_str = ""
  333. url = DEFAULT_URL
  334. config = ConfigParser.ConfigParser()
  335. config.read(CONFIG_FILES)
  336. # grab settings from config files
  337. if config.has_option('base', 'url'):
  338. url = config.get('base', 'url')
  339. if config.has_option('base', 'project'):
  340. project_str = config.get('base', 'project')
  341. for name, value in opts:
  342. if name == '-s':
  343. state_str = value
  344. elif name == '-p':
  345. project_str = value
  346. elif name == '-w':
  347. submitter_str = value
  348. elif name == '-d':
  349. delegate_str = value
  350. elif name == '-c':
  351. commit_str = value
  352. elif name == '-h':
  353. hash_str = value
  354. elif name == '-n':
  355. try:
  356. filt.add("max_count", int(value))
  357. except:
  358. sys.stderr.write("Invalid maximum count '%s'\n" % value)
  359. usage()
  360. else:
  361. sys.stderr.write("Unknown option '%s'\n" % name)
  362. usage()
  363. if len(args) > 1:
  364. sys.stderr.write("Too many arguments specified\n")
  365. usage()
  366. (username, password) = (None, None)
  367. transport = None
  368. if action in auth_actions:
  369. if config.has_option('auth', 'username') and \
  370. config.has_option('auth', 'password'):
  371. use_https = url.startswith('https')
  372. transport = BasicHTTPAuthTransport( \
  373. config.get('auth', 'username'),
  374. config.get('auth', 'password'),
  375. use_https)
  376. else:
  377. sys.stderr.write(("The %s action requires authentication, "
  378. "but no username or password\nis configured\n") % action)
  379. sys.exit(1)
  380. if project_str:
  381. filt.add("project", project_str)
  382. if state_str:
  383. filt.add("state", state_str)
  384. try:
  385. rpc = xmlrpclib.Server(url, transport = transport)
  386. except:
  387. sys.stderr.write("Unable to connect to %s\n" % url)
  388. sys.exit(1)
  389. patch_id = None
  390. if hash_str:
  391. patch_id = patch_id_from_hash(rpc, project_str, hash_str)
  392. if patch_id is None:
  393. sys.stderr.write("No patch has the hash provided\n")
  394. sys.exit(1)
  395. if action == 'list' or action == 'search':
  396. if len(args) > 0:
  397. filt.add("name__icontains", args[0])
  398. action_list(rpc, filt, submitter_str, delegate_str)
  399. elif action.startswith('project'):
  400. action_projects(rpc)
  401. elif action.startswith('state'):
  402. action_states(rpc)
  403. elif action == 'branch':
  404. try:
  405. patch_id = patch_id or int(args[0])
  406. except:
  407. sys.stderr.write("Invalid patch ID given\n")
  408. sys.exit(1)
  409. branch_with(patch_id, rpc)
  410. elif action == 'view':
  411. try:
  412. patch_id = patch_id or int(args[0])
  413. except:
  414. sys.stderr.write("Invalid patch ID given\n")
  415. sys.exit(1)
  416. s = rpc.patch_get_mbox(patch_id)
  417. if len(s) > 0:
  418. print unicode(s).encode("utf-8")
  419. elif action == 'get' or action == 'save':
  420. try:
  421. patch_id = patch_id or int(args[0])
  422. except:
  423. sys.stderr.write("Invalid patch ID given\n")
  424. sys.exit(1)
  425. action_get(rpc, patch_id)
  426. elif action == 'apply':
  427. try:
  428. patch_id = patch_id or int(args[0])
  429. except:
  430. sys.stderr.write("Invalid patch ID given\n")
  431. sys.exit(1)
  432. action_apply(rpc, patch_id)
  433. elif action == 'update':
  434. try:
  435. patch_id = patch_id or int(args[0])
  436. except:
  437. sys.stderr.write("Invalid patch ID given\n")
  438. sys.exit(1)
  439. action_update_patch(rpc, patch_id, state = state_str,
  440. commit = commit_str)
  441. else:
  442. sys.stderr.write("Unknown action '%s'\n" % action)
  443. usage()
  444. if __name__ == "__main__":
  445. main()