pyth_utils.py 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. import os
  2. import socketserver
  3. import subprocess
  4. import sys
  5. # Settings specific to local devnet Pyth instance
  6. PYTH = os.environ.get("PYTH", "./pyth")
  7. PYTH_ADMIN = os.environ.get("PYTH_ADMIN", "./pyth_admin")
  8. PYTH_KEY_STORE = os.environ.get("PYTH_KEY_STORE", "/home/pyth/.pythd")
  9. PYTH_PROGRAM_KEYPAIR = os.environ.get(
  10. "PYTH_PROGRAM_KEYPAIR", f"{PYTH_KEY_STORE}/publish_key_pair.json"
  11. )
  12. PYTH_PROGRAM_SO_PATH = os.environ.get("PYTH_PROGRAM_SO", "../target/oracle.so")
  13. PYTH_PUBLISHER_KEYPAIR = os.environ.get(
  14. "PYTH_PUBLISHER_KEYPAIR", f"{PYTH_KEY_STORE}/publish_key_pair.json"
  15. )
  16. PYTH_PUBLISHER_INTERVAL = float(os.environ.get("PYTH_PUBLISHER_INTERVAL", "5"))
  17. PYTH_MAPPING_KEYPAIR = os.environ.get(
  18. "PYTH_MAPPING_KEYPAIR", f"{PYTH_KEY_STORE}/mapping_key_pair.json"
  19. )
  20. # 0 setting disables airdropping
  21. SOL_AIRDROP_AMT = int(os.environ.get("SOL_AIRDROP_AMT", 0))
  22. # SOL RPC settings
  23. SOL_RPC_HOST = os.environ.get("SOL_RPC_HOST", "solana-devnet")
  24. SOL_RPC_PORT = int(os.environ.get("SOL_RPC_PORT", 8899))
  25. SOL_RPC_URL = os.environ.get(
  26. "SOL_RPC_URL", "http://{0}:{1}".format(SOL_RPC_HOST, SOL_RPC_PORT)
  27. )
  28. # A TCP port we open when a service is ready
  29. READINESS_PORT = int(os.environ.get("READINESS_PORT", "2000"))
  30. def run_or_die(args, die=True, **kwargs):
  31. """
  32. Opinionated subprocess.run() call with fancy logging
  33. """
  34. args_readable = " ".join(args)
  35. print(f"CMD RUN\t{args_readable}", file=sys.stderr)
  36. sys.stderr.flush()
  37. ret = subprocess.run(args, text=True, **kwargs)
  38. if ret.returncode != 0:
  39. print(f"CMD FAIL {ret.returncode}\t{args_readable}", file=sys.stderr)
  40. out = ret.stdout if ret.stdout is not None else "<not captured>"
  41. err = ret.stderr if ret.stderr is not None else "<not captured>"
  42. print(f"CMD STDOUT\n{out}", file=sys.stderr)
  43. print(f"CMD STDERR\n{err}", file=sys.stderr)
  44. if die:
  45. sys.exit(ret.returncode)
  46. else:
  47. print(f'{"CMD DIE FALSE"}', file=sys.stderr)
  48. else:
  49. print(f"CMD OK\t{args_readable}", file=sys.stderr)
  50. sys.stderr.flush()
  51. return ret
  52. def pyth_run_or_die(subcommand, args=[], debug=False, **kwargs):
  53. """
  54. Pyth boilerplate in front of run_or_die.
  55. """
  56. return run_or_die(
  57. [PYTH, subcommand] + args + (["-d"] if debug else [])
  58. + ["-k", PYTH_KEY_STORE]
  59. + ["-r", SOL_RPC_HOST]
  60. + ["-c", "finalized"]
  61. + ["-x"], # These means to bypass transaction proxy server. In this setup it's not running and it's required to bypass
  62. **kwargs,
  63. )
  64. def pyth_admin_run_or_die(subcommand, args=[], debug=False, **kwargs):
  65. """
  66. Pyth_admin boilerplate in front of run_or_die.
  67. """
  68. return run_or_die(
  69. [PYTH_ADMIN, subcommand] + args + (["-d"] if debug else [])
  70. + ["-n"] # These commands require y/n confirmation. This bypasses that
  71. + ["-k", PYTH_KEY_STORE]
  72. + ["-r", SOL_RPC_HOST]
  73. + ["-c", "finalized"],
  74. **kwargs,
  75. )
  76. def sol_run_or_die(subcommand, args=[], **kwargs):
  77. """
  78. Solana boilerplate in front of run_or_die
  79. """
  80. return run_or_die(["solana", subcommand] + args + ["--url", SOL_RPC_URL], **kwargs)
  81. class ReadinessTCPHandler(socketserver.StreamRequestHandler):
  82. def handle(self):
  83. """TCP black hole"""
  84. self.rfile.read(64)
  85. def readiness():
  86. """
  87. Accept connections from readiness probe
  88. """
  89. with socketserver.TCPServer(
  90. ("0.0.0.0", READINESS_PORT), ReadinessTCPHandler
  91. ) as srv:
  92. srv.serve_forever()