/scripts/api/common.py

https://bitbucket.org/cistrome/cistrome-harvard/ · Python · 198 lines · 151 code · 13 blank · 34 comment · 18 complexity · 4c5bea0bf70184e2fb44607840f5be4b MD5 · raw file

  1. """
  2. Common methods used by the API sample scripts.
  3. """
  4. import json
  5. import logging
  6. import os
  7. import sys
  8. import urllib2
  9. new_path = [ os.path.join( os.path.dirname( __file__ ), '..', '..', 'lib' ) ]
  10. new_path.extend( sys.path[1:] )
  11. sys.path = new_path
  12. from galaxy import eggs
  13. eggs.require( "pycrypto" )
  14. from Crypto.Cipher import Blowfish
  15. log = logging.getLogger( __name__ )
  16. def make_url( api_key, url, args=None ):
  17. """
  18. Adds the API Key to the URL if it's not already there.
  19. """
  20. if args is None:
  21. args = []
  22. argsep = '&'
  23. if '?' not in url:
  24. argsep = '?'
  25. if '?key=' not in url and '&key=' not in url:
  26. args.insert( 0, ( 'key', api_key ) )
  27. return url + argsep + '&'.join( [ '='.join( t ) for t in args ] )
  28. def get( api_key, url ):
  29. """
  30. Do the actual GET.
  31. """
  32. url = make_url( api_key, url )
  33. try:
  34. return json.loads( urllib2.urlopen( url ).read() )
  35. except ValueError, e:
  36. print "URL did not return JSON data: %s" % e
  37. sys.exit(1)
  38. def post( api_key, url, data ):
  39. """
  40. Do the actual POST.
  41. """
  42. url = make_url( api_key, url )
  43. req = urllib2.Request( url, headers = { 'Content-Type': 'application/json' }, data = json.dumps( data ) )
  44. return json.loads( urllib2.urlopen( req ).read() )
  45. def put( api_key, url, data ):
  46. """
  47. Do the actual PUT
  48. """
  49. url = make_url( api_key, url )
  50. req = urllib2.Request( url, headers = { 'Content-Type': 'application/json' }, data = json.dumps( data ))
  51. req.get_method = lambda: 'PUT'
  52. return json.loads( urllib2.urlopen( req ).read() )
  53. def __del( api_key, url, data ):
  54. """
  55. Do the actual DELETE
  56. """
  57. url = make_url( api_key, url )
  58. req = urllib2.Request( url, headers = { 'Content-Type': 'application/json' }, data = json.dumps( data ))
  59. req.get_method = lambda: 'DELETE'
  60. return json.loads( urllib2.urlopen( req ).read() )
  61. def display( api_key, url, return_formatted=True ):
  62. """
  63. Sends an API GET request and acts as a generic formatter for the JSON response.
  64. """
  65. try:
  66. r = get( api_key, url )
  67. except urllib2.HTTPError, e:
  68. print e
  69. print e.read( 1024 ) # Only return the first 1K of errors.
  70. sys.exit( 1 )
  71. if type( r ) == unicode:
  72. print 'error: %s' % r
  73. return None
  74. if not return_formatted:
  75. return r
  76. elif type( r ) == list:
  77. # Response is a collection as defined in the REST style.
  78. print 'Collection Members'
  79. print '------------------'
  80. for n, i in enumerate(r):
  81. # All collection members should have a name and url in the response.
  82. print '#%d: %s' % (n+1, i.pop( 'url' ) )
  83. if 'name' in i:
  84. print ' name: %s' % i.pop( 'name' )
  85. for k, v in i.items():
  86. print ' %s: %s' % ( k, v )
  87. print ''
  88. print '%d element(s) in collection' % len( r )
  89. elif type( r ) == dict:
  90. # Response is an element as defined in the REST style.
  91. print 'Member Information'
  92. print '------------------'
  93. for k, v in r.items():
  94. print '%s: %s' % ( k, v )
  95. elif type( r ) == str:
  96. print r
  97. else:
  98. print 'response is unknown type: %s' % type( r )
  99. def submit( api_key, url, data, return_formatted=True ):
  100. """
  101. Sends an API POST request and acts as a generic formatter for the JSON response.
  102. 'data' will become the JSON payload read by Galaxy.
  103. """
  104. try:
  105. r = post( api_key, url, data )
  106. except urllib2.HTTPError, e:
  107. if return_formatted:
  108. print e
  109. print e.read( 1024 )
  110. sys.exit( 1 )
  111. else:
  112. return 'Error. '+ str( e.read( 1024 ) )
  113. if not return_formatted:
  114. return r
  115. print 'Response'
  116. print '--------'
  117. if type( r ) == list:
  118. # Currently the only implemented responses are lists of dicts, because
  119. # submission creates some number of collection elements.
  120. for i in r:
  121. if type( i ) == dict:
  122. if 'url' in i:
  123. print i.pop( 'url' )
  124. else:
  125. print '----'
  126. if 'name' in i:
  127. print ' name: %s' % i.pop( 'name' )
  128. for k, v in i.items():
  129. print ' %s: %s' % ( k, v )
  130. else:
  131. print i
  132. else:
  133. print r
  134. def update( api_key, url, data, return_formatted=True ):
  135. """
  136. Sends an API PUT request and acts as a generic formatter for the JSON response.
  137. 'data' will become the JSON payload read by Galaxy.
  138. """
  139. try:
  140. r = put( api_key, url, data )
  141. except urllib2.HTTPError, e:
  142. if return_formatted:
  143. print e
  144. print e.read( 1024 )
  145. sys.exit( 1 )
  146. else:
  147. return 'Error. '+ str( e.read( 1024 ) )
  148. if not return_formatted:
  149. return r
  150. print 'Response'
  151. print '--------'
  152. print r
  153. def delete( api_key, url, data, return_formatted=True ):
  154. """
  155. Sends an API DELETE request and acts as a generic formatter for the JSON response.
  156. 'data' will become the JSON payload read by Galaxy.
  157. """
  158. try:
  159. r = __del( api_key, url, data )
  160. except urllib2.HTTPError, e:
  161. if return_formatted:
  162. print e
  163. print e.read( 1024 )
  164. sys.exit( 1 )
  165. else:
  166. return 'Error. '+ str( e.read( 1024 ) )
  167. if not return_formatted:
  168. return r
  169. print 'Response'
  170. print '--------'
  171. print r
  172. def encode_id( config_id_secret, obj_id ):
  173. """
  174. utility method to encode ID's
  175. """
  176. id_cipher = Blowfish.new( config_id_secret )
  177. # Convert to string
  178. s = str( obj_id )
  179. # Pad to a multiple of 8 with leading "!"
  180. s = ( "!" * ( 8 - len(s) % 8 ) ) + s
  181. # Encrypt
  182. return id_cipher.encrypt( s ).encode( 'hex' )