/scripts/api/common.py

https://bitbucket.org/thata/galaxy-central · Python · 172 lines · 135 code · 16 blank · 21 comment · 41 complexity · 57e11f7a1ff4ef8584cccb8a168025e1 MD5 · raw file

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