linker.py 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. import sys
  2. import urllib
  3. import urllib2
  4. import json
  5. def flush_data(address, data, username):
  6. if data:
  7. urllib2.urlopen(urllib2.Request(address, urllib.urlencode({"op": "set_input", "data": json.dumps(data), "username": username})), timeout=10).read()
  8. return []
  9. def link(address, username, objects):
  10. # Read out all symbol tables that are to be linked
  11. definers = {}
  12. users = {}
  13. data = []
  14. definers["main"] = None
  15. for obj in objects:
  16. data.append(("V", '3'))
  17. data.append(("V", '"read_symbols"'))
  18. data.append(("V", '"%s"' % obj))
  19. data = flush_data(address, data, username)
  20. for obj in objects:
  21. print("[SYMB] %s" % (obj))
  22. v = urllib2.urlopen(urllib2.Request(address, urllib.urlencode({"op": "get_output", "username": username}))).read()
  23. lst = v.rsplit("=", 1)[1]
  24. lst = lst.split("\n")
  25. for e in lst:
  26. if len(e) > 1:
  27. name, defined = e.rsplit(":", 1)
  28. if defined == "1":
  29. if definers.get(name, None):
  30. raise Exception("Double definition for symbol %s\nDefined in %s\nDefined in %s" % (name, definers[name], obj))
  31. definers[name] = obj
  32. else:
  33. users.setdefault(name, []).append(obj)
  34. if name not in definers:
  35. definers[name] = None
  36. # Check for undefined symbols with this linking set
  37. for symbol in definers:
  38. if definers[symbol] is None:
  39. if symbol not in ["input", "output"]:
  40. # Some symbols are built-ins which only look like functions
  41. raise Exception("Undefined symbol %s.\nUsed by modules:\n\t%s" % (symbol, "\n\t".join(users[symbol])))
  42. # Ok, we know that all symbols can be defined with this set of files, now link their initializers together
  43. initializers = []
  44. for obj in objects:
  45. data.append(("V", '3'))
  46. data.append(("V", '"read_initializers"'))
  47. data.append(("V", '"%s"' % obj))
  48. data = flush_data(address, data, username)
  49. for obj in objects:
  50. print("[LINK] %s" % (obj))
  51. v = urllib2.urlopen(urllib2.Request(address, urllib.urlencode({"op": "get_output", "username": username}))).read()
  52. start = str(v.split("&", 1)[0].split("=")[1])
  53. initializers.append(start)
  54. # Bind all initializers together
  55. if definers:
  56. print("[LOAD] %s:main()" % definers["main"])
  57. else:
  58. print("[LOAD] main()")
  59. # Set interface to constructors
  60. commands = [("V", '1')]
  61. # Link all initializers together
  62. for init in initializers:
  63. commands.extend([
  64. ("V", '"call"'),
  65. ("V", '"access"'),
  66. ("V", '"resolve"'),
  67. ("V", '"exec"'),
  68. ("V", '1'),
  69. ("V", '"const"'),
  70. ("R", str(init)),
  71. ("V", 'true'),
  72. ])
  73. # Load main function
  74. commands.extend([
  75. ("V", '"return"'),
  76. ("V", 'true'),
  77. ("V", '"call"'),
  78. ("V", '"access"'),
  79. ("V", '"resolve"'),
  80. ("V", '"main"'),
  81. ("V", '0'),
  82. ("V", 'false'),
  83. ])
  84. import json
  85. urllib2.urlopen(urllib2.Request(address, urllib.urlencode({"op": "set_input", "data": json.dumps(commands), "username": username}))).read()
  86. if __name__ == "__main__":
  87. if len(sys.argv) == 1:
  88. print("No username defined")
  89. else:
  90. address = sys.argv[1]
  91. username = sys.argv[2]
  92. objects = set(sys.argv[3:])
  93. if objects:
  94. link(address, username, objects)
  95. else:
  96. print("No files to link defined")