/lib/tool_shed/scripts/api/common.py

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