pyth_utils.py 3.4 KB

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