linker.py 3.9 KB

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