flexible.py 121 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997
  1. from twisted.internet import reactor
  2. from twisted.internet import task
  3. from twisted.internet import defer
  4. from colorama import Fore, Back, Style
  5. # from twisted.python import log
  6. from twisted.internet.task import coiterate
  7. from twisted.internet.defer import gatherResults
  8. from itertools import cycle
  9. import pendulum
  10. from pprint import pformat
  11. from galaxy import GameData, PORT_CLASSES, CLASSES_PORT
  12. from boxes import Boxes
  13. import logging
  14. BLINK = '\x1b[5m'
  15. log = logging.getLogger(__name__)
  16. class SpinningCursor(object):
  17. """ Spinner class, that handles every so many clicks
  18. s = SpinningCursor(5) # every 5
  19. for x in range(10):
  20. if s.click():
  21. print(s.cycle())
  22. """
  23. def __init__(self, every=10):
  24. self.itercycle = cycle(["/", "-", "\\", "|"])
  25. self.count = 0
  26. self.every = every
  27. def reset(self):
  28. self.itercycle = cycle(["/", "-", "\\", "|"])
  29. self.count = 0
  30. def click(self):
  31. self.count += 1
  32. return self.count % self.every == 0
  33. def cycle(self):
  34. return next(self.itercycle)
  35. def again(self, count: int):
  36. return self.count % count == 0
  37. def merge(color_string):
  38. """ Given a string of colorama ANSI, merge them if you can. """
  39. return color_string.replace("m\x1b[", ";")
  40. class PlayerInput(object):
  41. """ Player Input
  42. Example:
  43. from flexible import PlayerInput
  44. ask = PlayerInput(self.game)
  45. # abort_blank means, if the input field is blank, abort. Use error_back.
  46. d = ask.prompt("What is your quest?", 40, name="quest", abort_blank=True)
  47. # Display the user's input / but not needed.
  48. d.addCallback(ask.output)
  49. d.addCallback(
  50. lambda ignore: ask.prompt(
  51. "What is your favorite color?", 10, name="color"
  52. )
  53. )
  54. d.addCallback(ask.output)
  55. d.addCallback(
  56. lambda ignore: ask.prompt(
  57. "What is your least favorite number?",
  58. 12,
  59. name="number",
  60. digits=True,
  61. )
  62. )
  63. d.addCallback(ask.output)
  64. def show_values(show):
  65. log.debug(show)
  66. self.queue_game.put(pformat(show).replace("\n", "\n\r") + self.nl)
  67. d.addCallback(lambda ignore: show_values(ask.keep))
  68. d.addCallback(self.welcome_back)
  69. # On error, just return back
  70. d.addErrback(self.welcome_back)
  71. """
  72. def __init__(self, game):
  73. # I think game gives us access to everything we need
  74. self.game = game
  75. self.observer = self.game.observer
  76. self.save = None
  77. self.deferred = None
  78. self.queue_game = game.queue_game
  79. self.keep = {}
  80. # default colors
  81. # prompt
  82. self.c = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  83. # input
  84. self.cp = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  85. # useful constants
  86. self.r = Style.RESET_ALL
  87. self.nl = "\n\r"
  88. self.bsb = "\b \b"
  89. self.keepalive = None
  90. def color(self, c):
  91. self.c = c
  92. def colorp(self, cp):
  93. self.cp = cp
  94. def alive(self):
  95. log.debug("PlayerInput.alive()")
  96. self.game.queue_player.put(" ")
  97. def prompt(self, user_prompt, limit, **kw):
  98. """ Generate prompt for user input.
  99. Note: This returns deferred.
  100. prompt = text displayed.
  101. limit = # of characters allowed.
  102. default = (text to default to)
  103. keywords:
  104. abort_blank : Abort if they give us blank text.
  105. name : Stores the input in self.keep dict.
  106. digits : Only allow 0-9 to be entered.
  107. """
  108. log.debug("PlayerInput({0}, {1}, {2}".format(user_prompt, limit, kw))
  109. self.limit = limit
  110. self.input = ""
  111. self.kw = kw
  112. assert self.save is None
  113. assert self.keepalive is None
  114. # Note: This clears out the server "keep alive"
  115. self.save = self.observer.save()
  116. self.observer.connect("player", self.get_input)
  117. self.keepalive = task.LoopingCall(self.alive)
  118. self.keepalive.start(30)
  119. # We need to "hide" the game output.
  120. # Otherwise it WITH mess up the user input display.
  121. self.to_player = self.game.to_player
  122. self.game.to_player = False
  123. # Display prompt
  124. # self.queue_game.put(self.r + self.nl + self.c + user_prompt + " " + self.cp)
  125. self.queue_game.put(self.r + self.c + user_prompt + self.r + " " + self.cp)
  126. # Set "Background of prompt"
  127. self.queue_game.put(" " * limit + "\b" * limit)
  128. assert self.deferred is None
  129. d = defer.Deferred()
  130. self.deferred = d
  131. log.debug("Return deferred ...")
  132. return d
  133. def get_input(self, chunk):
  134. """ Data from player (in bytes) """
  135. chunk = chunk.decode("latin-1", "ignore")
  136. for ch in chunk:
  137. if ch == "\b":
  138. if len(self.input) > 0:
  139. self.queue_game.put(self.bsb)
  140. self.input = self.input[0:-1]
  141. else:
  142. self.queue_game.put("\a")
  143. elif ch == "\r":
  144. self.queue_game.put(self.r + self.nl)
  145. log.debug("Restore observer dispatch {0}".format(self.save))
  146. assert not self.save is None
  147. self.observer.load(self.save)
  148. self.save = None
  149. log.debug("Disable keepalive")
  150. self.keepalive.stop()
  151. self.keepalive = None
  152. line = self.input
  153. self.input = ""
  154. assert not self.deferred is None
  155. self.game.to_player = self.to_player
  156. # If they gave us the keyword name, save the value as that name
  157. if "name" in self.kw:
  158. self.keep[self.kw["name"]] = line
  159. if "abort_blank" in self.kw and self.kw["abort_blank"]:
  160. # Abort on blank input
  161. if line.strip() == "":
  162. # Yes, input is blank, abort.
  163. log.info("errback, abort_blank")
  164. reactor.callLater(
  165. 0, self.deferred.errback, Exception("abort_blank")
  166. )
  167. self.deferred = None
  168. return
  169. # Ok, use deferred.callback, or reactor.callLater?
  170. # self.deferred.callback(line)
  171. reactor.callLater(0, self.deferred.callback, line)
  172. self.deferred = None
  173. return
  174. elif ch.isprintable():
  175. # Printable, but is it acceptable?
  176. if "digits" in self.kw:
  177. if not ch.isdigit():
  178. self.queue_game.put("\a")
  179. continue
  180. if len(self.input) + 1 <= self.limit:
  181. self.input += ch
  182. self.queue_game.put(ch)
  183. else:
  184. self.queue_game.put("\a")
  185. def output(self, line):
  186. """ A default display of what they just input. """
  187. log.debug("PlayerInput.output({0})".format(line))
  188. self.game.queue_game.put(self.r + "[{0}]".format(line) + self.nl)
  189. return line
  190. import re
  191. # The CIMWarpReport -- is only needed if the json file gets damaged in some way.
  192. # Like when the universe gets bigbanged. :P
  193. # or needs to be reset. The warps should automatically update themselves now.
  194. class CIMWarpReport(object):
  195. def __init__(self, game):
  196. self.game = game
  197. self.queue_game = game.queue_game
  198. self.queue_player = game.queue_player
  199. self.observer = game.observer
  200. # Yes, at this point we would activate
  201. self.prompt = game.buffer
  202. self.save = self.observer.save()
  203. # I actually don't want the player input, but I'll grab it anyway.
  204. self.observer.connect("player", self.player)
  205. self.observer.connect("prompt", self.game_prompt)
  206. self.observer.connect("game-line", self.game_line)
  207. # If we want it, it's here.
  208. self.defer = None
  209. self.to_player = self.game.to_player
  210. # Hide what's happening from the player
  211. self.game.to_player = False
  212. self.queue_player.put("^") # Activate CIM
  213. self.state = 1
  214. # self.warpdata = {}
  215. self.queue_game.put("Loading ... ") # cycle eats last char.
  216. self.warpcycle = SpinningCursor()
  217. def game_prompt(self, prompt):
  218. if prompt == ": ":
  219. if self.state == 1:
  220. # Ok, then we're ready to request the port report
  221. self.warpcycle.reset()
  222. self.queue_player.put("I")
  223. self.state = 2
  224. elif self.state == 2:
  225. self.queue_player.put("Q")
  226. self.state = 3
  227. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  228. if self.state == 3:
  229. # Ok, time to exit
  230. # exit from this...
  231. self.game.to_player = self.to_player
  232. self.observer.load(self.save)
  233. self.save = None
  234. # self.game.warpdata = self.warpdata
  235. self.queue_game.put("\b \b\r\n")
  236. if not self.defer is None:
  237. self.defer.callback(self.game.gamedata.warps)
  238. self.defer = None
  239. def game_line(self, line):
  240. if line == "" or line == ": ":
  241. return
  242. if line == ": ENDINTERROG":
  243. return
  244. if line.startswith('Command [TL='):
  245. return
  246. # This should be the CIM Report Data -- parse it
  247. if self.warpcycle:
  248. if self.warpcycle.click():
  249. self.queue_game.put("\b")
  250. if self.warpcycle.again(1000):
  251. self.queue_game.put(".")
  252. self.queue_game.put(self.warpcycle.cycle())
  253. work = line.strip()
  254. parts = re.split(r"(?<=\d)\s", work)
  255. parts = [int(x) for x in parts]
  256. sector = parts.pop(0)
  257. # tuples are nicer on memory, and the warpdata map isn't going to be changing.
  258. # self.warpdata[sector] = tuple(parts)
  259. self.game.gamedata.warp_to(sector, *parts)
  260. def __del__(self):
  261. log.debug("CIMWarpReport {0} RIP".format(self))
  262. def whenDone(self):
  263. self.defer = defer.Deferred()
  264. # Call this to chain something after we exit.
  265. return self.defer
  266. def player(self, chunk):
  267. """ Data from player (in bytes). """
  268. chunk = chunk.decode("latin-1", "ignore")
  269. key = chunk.upper()
  270. log.warn("CIMWarpReport.player({0}) : I AM stopping...".format(key))
  271. # Stop the keepalive if we are activating something else
  272. # or leaving...
  273. # self.keepalive.stop()
  274. self.queue_game.put("\b \b\r\n")
  275. if not self.defer is None:
  276. # We have something, so:
  277. self.game.to_player = self.to_player
  278. self.observer.load(self.save)
  279. self.save = None
  280. self.defer.errback(Exception("User Abort"))
  281. self.defer = None
  282. else:
  283. # Still "exit" out.
  284. self.game.to_player = self.to_player
  285. self.observer.load(self.save)
  286. # the CIMPortReport will still be needed.
  287. # We can't get fresh report data (that changes) any other way.
  288. class CIMPortReport(object):
  289. """ Parse data from CIM Port Report
  290. Example:
  291. from flexible import CIMPortReport
  292. report = CIMPortReport(self.game)
  293. d = report.whenDone()
  294. d.addCallback(self.port_report)
  295. d.addErrback(self.welcome_back)
  296. def port_report(self, portdata):
  297. self.portdata = portdata
  298. self.queue_game.put("Loaded {0} records.".format(len(portdata)) + self.nl)
  299. self.welcome_back()
  300. def welcome_back(self,*_):
  301. ... restore keep alive timers, etc.
  302. """
  303. def __init__(self, game):
  304. self.game = game
  305. self.queue_game = game.queue_game
  306. self.queue_player = game.queue_player
  307. self.observer = game.observer
  308. # Yes, at this point we would activate
  309. self.prompt = game.buffer
  310. self.save = self.observer.save()
  311. # I actually don't want the player input, but I'll grab it anyway.
  312. self.observer.connect("player", self.player)
  313. self.observer.connect("prompt", self.game_prompt)
  314. self.observer.connect("game-line", self.game_line)
  315. # If we want it, it's here.
  316. self.defer = None
  317. self.to_player = self.game.to_player
  318. log.debug("to_player (stored) {0}".format(self.to_player))
  319. # Hide what's happening from the player
  320. self.game.to_player = False
  321. self.queue_player.put("^") # Activate CIM
  322. self.state = 1
  323. # self.portdata = {}
  324. self.queue_game.put("Loading ... ") # cycle eats last char.
  325. self.portcycle = SpinningCursor()
  326. def game_prompt(self, prompt):
  327. if prompt == ": ":
  328. if self.state == 1:
  329. # Ok, then we're ready to request the port report
  330. self.portcycle.reset()
  331. self.queue_player.put("R")
  332. self.state = 2
  333. elif self.state == 2:
  334. self.queue_player.put("Q")
  335. self.state = 3
  336. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  337. if self.state == 3:
  338. # Ok, time to exit
  339. # exit from this...
  340. self.game.to_player = self.to_player
  341. self.observer.load(self.save)
  342. self.save = None
  343. # self.game.portdata = self.portdata
  344. self.queue_game.put("\b \b\r\n")
  345. if not self.defer is None:
  346. self.defer.callback(self.game.gamedata.ports)
  347. self.defer = None
  348. def game_line(self, line: str):
  349. if line == "" or line == ": ":
  350. return
  351. if line == ": ENDINTERROG":
  352. return
  353. # This should be the CIM Report Data -- parse it
  354. if self.portcycle:
  355. if self.portcycle.click():
  356. self.queue_game.put("\b")
  357. if self.portcycle.again(1000):
  358. self.queue_game.put(".")
  359. self.queue_game.put(self.portcycle.cycle())
  360. work = line.replace("%", "")
  361. parts = re.split(r"(?<=\d)\s", work)
  362. if len(parts) == 8:
  363. port = int(parts[0].strip())
  364. data = dict()
  365. def portBS(info):
  366. if info[0] == "-":
  367. bs = "B"
  368. else:
  369. bs = "S"
  370. return (bs, int(info[1:].strip()))
  371. data["fuel"] = dict()
  372. data["fuel"]["sale"], data["fuel"]["units"] = portBS(parts[1])
  373. data["fuel"]["pct"] = int(parts[2].strip())
  374. data["org"] = dict()
  375. data["org"]["sale"], data["org"]["units"] = portBS(parts[3])
  376. data["org"]["pct"] = int(parts[4].strip())
  377. data["equ"] = dict()
  378. data["equ"]["sale"], data["equ"]["units"] = portBS(parts[5])
  379. data["equ"]["pct"] = int(parts[6].strip())
  380. # Store what this port is buying/selling
  381. data["port"] = (
  382. data["fuel"]["sale"] + data["org"]["sale"] + data["equ"]["sale"]
  383. )
  384. # Convert BBS/SBB to Class number 1-8
  385. data["class"] = CLASSES_PORT[data["port"]]
  386. self.game.gamedata.set_port(port, data)
  387. # self.portdata[port] = data
  388. else:
  389. log.error("CIMPortReport: {0} ???".format(line))
  390. def __del__(self):
  391. log.debug("CIMPortReport {0} RIP".format(self))
  392. def whenDone(self):
  393. self.defer = defer.Deferred()
  394. # Call this to chain something after we exit.
  395. return self.defer
  396. def player(self, chunk):
  397. """ Data from player (in bytes). """
  398. chunk = chunk.decode("latin-1", "ignore")
  399. key = chunk.upper()
  400. log.warn("CIMPortReport.player({0}) : I AM stopping...".format(key))
  401. # Stop the keepalive if we are activating something else
  402. # or leaving...
  403. # self.keepalive.stop()
  404. self.queue_game.put("\b \b\r\n")
  405. if not self.defer is None:
  406. # We have something, so:
  407. self.game.to_player = self.to_player
  408. self.observer.load(self.save)
  409. self.save = None
  410. self.defer.errback(Exception("User Abort"))
  411. self.defer = None
  412. else:
  413. # Still "exit" out.
  414. self.game.to_player = self.to_player
  415. self.observer.load(self.save)
  416. class ScriptPort(object):
  417. """ Performs the Port script.
  418. This is close to the original.
  419. We don't ask for the port to trade with --
  420. because that information is available to us after "D" (display).
  421. We look at the adjacent sectors, and see if we know any ports.
  422. If the ports are burnt (< 20%), we remove them from the list.
  423. We sort the best trades first.
  424. If there's just one, we use it. Otherwise we ask them to choose.
  425. We have options Trade_UseFirst, which uses the first one, if
  426. there is more then one.
  427. Option Trade_Turns, will use this as default turns, without
  428. asking.
  429. """
  430. def __init__(self, game):
  431. self.game = game
  432. self.queue_game = game.queue_game
  433. self.queue_player = game.queue_player
  434. self.observer = game.observer
  435. self.r = Style.RESET_ALL
  436. self.nl = "\n\r"
  437. self.this_sector = None # Starting sector
  438. self.sector1 = None # Current Sector
  439. self.sector2 = None # Next Sector Stop
  440. self.percent = self.game.gamedata.get_config('Trade_Percent', '5')
  441. self.stop = self.game.gamedata.get_config('Trade_Stop', '10')
  442. try:
  443. self.stop = int(self.stop)
  444. except ValueError:
  445. self.stop = 10
  446. # Validate what we got from the config.
  447. # Not an int: make it 5, and update.
  448. # > 50, make it 5 and update.
  449. # < 0, make it 0 and update.
  450. update_config = False
  451. try:
  452. self.percent = int(self.percent)
  453. except ValueError:
  454. self.percent = 5
  455. update_config = True
  456. if self.percent > 50:
  457. self.percent = 5
  458. update_config = True
  459. if self.percent < 0:
  460. self.percent = 0
  461. update_config = True
  462. if update_config:
  463. self.game.gamedata.set_config('Trade_Percent', self.percent)
  464. self.credits = 0
  465. self.last_credits = None
  466. self.times_left = 0
  467. # Activate
  468. self.prompt = game.buffer
  469. self.save = self.observer.save()
  470. self.observer.connect('player', self.player)
  471. self.observer.connect("prompt", self.game_prompt)
  472. self.observer.connect("game-line", self.game_line)
  473. self.defer = None
  474. self.send2player(self.r + "Script based on: Port Pair Trading v2.00" + self.nl)
  475. self.possible_sectors = None
  476. self.state = 1
  477. self.queue_player.put("D")
  478. # Original, send 'D' to display current sector.
  479. # We could get the sector number from the self.prompt string -- HOWEVER:
  480. # IF! We send 'D', we can also get the sectors around -- we might not even need to
  481. # prompt for sector to trade with (we could possibly figure it out ourselves).
  482. # [Command [TL=00:00:00]:[967] (?=Help)? : D]
  483. # [<Re-Display>]
  484. # []
  485. # [Sector : 967 in uncharted space.]
  486. # [Planets : (M) Into the Darkness]
  487. # [Warps to Sector(s) : 397 - (562) - (639)]
  488. # []
  489. def whenDone(self):
  490. self.defer = defer.Deferred()
  491. # Call this to chain something after we exit.
  492. return self.defer
  493. def deactivate(self, andExit=True):
  494. self.state = 0
  495. log.debug("ScriptPort.deactivate ({0})".format(self.times_left))
  496. self.queue_game.put(self.nl + Boxes.alert("Trading Script deactivating..."))
  497. assert(not self.save is None)
  498. self.observer.load(self.save)
  499. self.save = None
  500. if self.defer:
  501. if andExit:
  502. self.defer.callback({'exit':True})
  503. else:
  504. self.defer.callback('done')
  505. self.defer = None
  506. def player(self, chunk: bytes):
  507. # If we receive anything -- ABORT!
  508. self.deactivate()
  509. def send2game(self, txt):
  510. log.debug("ScriptPort.send2game({0})".format(txt))
  511. self.queue_player.put(txt)
  512. def send2player(self, txt):
  513. log.debug("ScriptPort.send2player({0})".format(txt))
  514. self.queue_game.put(txt)
  515. def game_prompt(self, prompt: str):
  516. log.debug("{0} : {1}".format(self.state, prompt))
  517. if self.state == 3:
  518. # log.("game_prompt: ", prompt)
  519. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  520. self.state = 4
  521. log.debug("Ok, state 4")
  522. use_first = self.game.gamedata.get_config('Trade_UseFirst', 'N').upper()[0] == 'Y'
  523. if self.sector2 is None and use_first:
  524. # Use the first one by default
  525. self.sector2 = self.possible[0]
  526. log.info("default to {0}".format(self.sector2))
  527. if self.sector2 is None:
  528. # Ok, we need to prompt for this.
  529. self.queue_game.put(self.r + self.nl +
  530. "Which sector to trade with? {0}".format(GameData.port_show_part(self.this_sector, self.game.gamedata.ports[self.this_sector])) +
  531. self.nl)
  532. for i, p in enumerate(self.possible):
  533. self.queue_game.put(" " + Fore.CYAN + str(i + 1) + " : " + GameData.port_show_part(p, self.game.gamedata.ports[p]) + self.nl)
  534. pi = PlayerInput(self.game)
  535. def got_need1(*_):
  536. log.debug("Ok, I have: {0}".format(pi.keep))
  537. if pi.keep['count'].strip() == '':
  538. self.deactivate()
  539. return
  540. self.times_left = int(pi.keep['count'])
  541. if pi.keep['choice'].strip() == '':
  542. self.deactivate()
  543. return
  544. c = int(pi.keep['choice']) -1
  545. if c < 0 or c >= len(self.possible):
  546. self.deactivate()
  547. return
  548. self.sector2 = self.possible[int(pi.keep['choice']) -1]
  549. # self.queue_game.put(pformat(pi.keep).replace("\n", "\n\r"))
  550. self.state = 5
  551. self.trade()
  552. d = pi.prompt("Choose -=>", 5, name='choice', digits=True)
  553. d.addCallback(lambda ignore: pi.prompt("Times to execute script:", 5, name='count', digits=True))
  554. d.addCallback(got_need1)
  555. else:
  556. # We already have our target port, so...
  557. self.queue_game.put(self.r + self.nl + "Trading from {0} ({1}) to default {2} ({3}).".format(
  558. self.this_sector,
  559. self.game.gamedata.ports[self.this_sector]['port'],
  560. self.sector2, self.game.gamedata.ports[self.sector2]['port']) + self.nl
  561. )
  562. self.queue_game.put(self.r + self.nl + "Trading {0}".format(
  563. self.game.gamedata.port_trade_show(self.this_sector,
  564. self.sector2, 0)
  565. ))
  566. pi = PlayerInput(self.game)
  567. def got_need2(*_):
  568. if pi.keep['count'].strip() == '':
  569. self.deactivate()
  570. return
  571. self.times_left = int(pi.keep['count'])
  572. log.debug("Ok, I have: {0}".format(pi.keep))
  573. # self.queue_game.put(pformat(pi.keep).replace("\n", "\n\r"))
  574. self.state = 5
  575. self.trade()
  576. self.queue_game.put(self.r + self.nl)
  577. default_turns = self.game.gamedata.get_config('Trade_Turns', '0')
  578. if default_turns == '0':
  579. # No default given, ask.
  580. d = pi.prompt("Times to execute script", 5, name='count')
  581. d.addCallback(got_need2)
  582. else:
  583. try:
  584. self.times_left = int(default_turns)
  585. except ValueError:
  586. self.times_left = 30
  587. self.state = 5
  588. self.trade()
  589. elif self.state == 6:
  590. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  591. if self.end_trans:
  592. self.deactivate()
  593. return
  594. if self.fixable:
  595. # self.queue_game.put("Ok! Let's fix this by going to the other sector..." + self.nl)
  596. self.queue_game.put(self.nl + Boxes.alert("Ok, FINE. We'll trade with the other port.", base="green", style=0))
  597. log.debug("Fixing...")
  598. # Swap this and other sector
  599. self.this_sector, self.other_sector = (self.other_sector, self.this_sector)
  600. self.queue_player.put("{0}\r".format(self.sector2))
  601. self.state = 5
  602. self.trade()
  603. elif self.state == 7:
  604. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  605. # Done
  606. if self.end_trans:
  607. self.deactivate()
  608. return
  609. # Swap this and other sector
  610. self.this_sector, self.other_sector = (self.other_sector, self.this_sector)
  611. if self.this_sector == self.sector2:
  612. self.times_left -= 1
  613. if self.times_left <= 0:
  614. # Ok, exit out
  615. self.deactivate()
  616. return
  617. if self.last_credits is None:
  618. self.last_credits = self.credits
  619. else:
  620. if self.credits <= self.last_credits:
  621. log.warn("We don't seem to be making any money here...")
  622. self.queue_game.put(self.r + self.nl + "We don't seem to be making any money here. I'm stopping!" + self.nl)
  623. self.deactivate()
  624. return
  625. self.queue_player.put("{0}\r".format(self.this_sector))
  626. self.state = 10
  627. elif re.match(r'Your offer \[\d+\] \?', prompt):
  628. log.info("Your offer? [{0}]".format(self.fix_offer))
  629. if self.fix_offer:
  630. # Make real offer / WHAT?@?!
  631. work = prompt.replace(',', '')
  632. parts = re.split(r"\s+", work)
  633. amount = parts[2]
  634. # Ok, we have the amount, now to figure pct...
  635. if self.sell_percent > 100:
  636. self.sell_percent -= 1
  637. else:
  638. self.sell_percent += 1
  639. price = amount * self.sell_percent // 100
  640. log.debug("start: {0} % {1} price {2}".format(amount, self.sell_percent, price))
  641. if self.sell_percent > 100:
  642. self.sell_percent -= 1
  643. else:
  644. self.sell_percent += 1
  645. self.queue_player.put("{0}\r".format(price))
  646. # elif re.match(r"How many holds of .+ do you want to sell \[\d+\]\?", prompt):
  647. # log.info("Sell everything we can...")
  648. # this seems to screw up the sync of everything.
  649. # self.queue_player.put("\r")
  650. elif self.state == 8:
  651. # What are we trading
  652. # How many holds of Equipment do you want to buy [75]?
  653. if re.match(r"How many holds of .+ do you want to buy \[\d+\]\?", prompt):
  654. parts = prompt.split()
  655. trade_type = parts[4]
  656. log.info("Buy {0} [{1} ~ {2}]".format(trade_type, self.tpc, self.opc))
  657. if trade_type == 'Fuel':
  658. if (self.tpc in (5,7)) and (self.opc in (2,3,4,8)):
  659. # Can buy equipment - fuel ore is worthless.
  660. self.queue_player.put("0\r")
  661. return
  662. if (self.tpc in(4,7)) and (self.opc in (1,3,5,8)):
  663. # Can buy organics - fuel ore is worthless.
  664. self.queue_player.put("0\r")
  665. return
  666. if (self.tpc in (4,7,3,5)) and (self.opc in (3,4,5,7)):
  667. # No point in buying fuel ore if it can't be sold.
  668. self.queue_player.put("0\r")
  669. return
  670. elif trade_type == 'Organics':
  671. if (self.tpc in (6,7)) and (self.opc in (2,3,4,8)):
  672. # Can buy equipment - organics is worthless.
  673. self.queue_player.put("0\r")
  674. return
  675. if (self.tpc in (2,4,6,7)) and (self.opc in (2,4,6,7)):
  676. # No point in buying organics if it can't be sold.
  677. self.queue_player.put("0\r")
  678. return
  679. elif trade_type == 'Equipment':
  680. if (self.opc in (1,5,6,7)):
  681. # No point in buying equipment if it can't be sold.
  682. self.queue_player.put("0\r")
  683. return
  684. self.queue_player.put("\r")
  685. elif re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  686. # Done
  687. if self.end_trans:
  688. self.deactivate()
  689. return
  690. # Swap this and other sector
  691. self.this_sector, self.other_sector = (self.other_sector, self.this_sector)
  692. if self.this_sector == self.sector2:
  693. self.times_left -= 1
  694. if self.times_left <= 0:
  695. # Ok, exit out
  696. self.deactivate()
  697. return
  698. if self.last_credits is None:
  699. self.last_credits = self.credits
  700. else:
  701. if self.credits <= self.last_credits:
  702. log.warn("We don't seem to be making any money here...")
  703. self.queue_game.put(self.r + self.nl + "We don't seem to be making any money here. I'm stopping!" + self.nl)
  704. self.deactivate()
  705. return
  706. self.queue_player.put("{0}\r".format(self.this_sector))
  707. self.state = 10
  708. elif self.state == 99:
  709. # This is a good place to deactivate at.
  710. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  711. if hasattr(self, 'message'):
  712. if self.message is not None:
  713. self.queue_game.put(self.message)
  714. self.message = None
  715. self.deactivate()
  716. def trade(self, *_):
  717. # state 5
  718. log.debug("trade!")
  719. self.queue_player.put("pt") # Port Trade
  720. self.end_trans = False
  721. self.fixable = False
  722. self.this_port = self.game.gamedata.ports[self.this_sector]
  723. # I think other_sector will alway be correct, but leaving this
  724. # for now. FUTURE: TODO: REMOVE
  725. if self.this_sector == self.sector1:
  726. self.other_sector = self.sector2
  727. else:
  728. self.other_sector = self.sector1
  729. self.other_port = self.game.gamedata.ports[self.other_sector]
  730. # Ok, perform some calculations
  731. self.tpc = self.this_port['class']
  732. self.opc = self.other_port['class']
  733. self.fixable = 0
  734. self.fix_offer = 0
  735. # [ Items Status Trading % of max OnBoard]
  736. # [ ----- ------ ------- -------- -------]
  737. # [Fuel Ore Selling 2573 93% 0]
  738. # [Organics Buying 2960 100% 0]
  739. # [Equipment Buying 1958 86% 0]
  740. # []
  741. # []
  742. # [You have 1,000 credits and 20 empty cargo holds.]
  743. # []
  744. # [We are selling up to 2573. You have 0 in your holds.]
  745. # [How many holds of Fuel Ore do you want to buy [20]? 0]
  746. def game_line(self, line: str):
  747. if line.startswith("You have ") and 'credits and' in line:
  748. parts = line.replace(',', '').split()
  749. credits = int(parts[2])
  750. log.debug("Credits: {0}".format(credits))
  751. self.credits = credits
  752. if self.state == 1:
  753. # First exploration
  754. if line.startswith("Sector :"):
  755. # We have starting sector information
  756. parts = re.split(r"\s+", line)
  757. self.this_sector = int(parts[2])
  758. # These will be the ones swapped around as we trade back and forth.
  759. self.sector1 = self.this_sector
  760. elif line.startswith("Warps to Sector(s) : "):
  761. # Warps to Sector(s) : 397 - (562) - (639)
  762. _, _, warps = line.partition(':')
  763. warps = warps.replace('-', '').replace('(', '').replace(')', '').strip()
  764. log.debug("Warps: [{0}]".format(warps))
  765. self.warps = [ int(x) for x in re.split(r"\s+", warps)]
  766. log.debug("Warps: [{0}]".format(self.warps))
  767. self.state = 2
  768. elif self.state == 2:
  769. if line == "":
  770. # Ok, we're done
  771. self.state = 3
  772. # Check to see if we have information on any possible ports
  773. # if hasattr(self.game, 'portdata'):
  774. if True:
  775. if not self.this_sector in self.game.gamedata.ports:
  776. self.state = 0
  777. log.debug("Current sector {0} not in portdata.".format(self.this_sector))
  778. self.queue_game.put(self.r + self.nl + "I can't find the current sector in the portdata." + self.nl)
  779. self.deactivate()
  780. return
  781. else:
  782. # Ok, we are in the portdata
  783. pd = self.game.gamedata.ports[self.this_sector]
  784. if GameData.port_burnt(pd):
  785. log.debug("Current sector {0} port is burnt (<= 20%).".format(self.this_sector))
  786. self.queue_game.put(self.r + self.nl + "Current sector port is burnt out. <= 20%." + self.nl)
  787. self.deactivate()
  788. return
  789. possible = [ x for x in self.warps if x in self.game.gamedata.ports ]
  790. log.debug("Possible: {0}".format(possible))
  791. # BUG: Sometimes links to another sector, don't link back!
  792. # This causes the game to plot a course / autopilot.
  793. # if hasattr(self.game, 'warpdata'):
  794. if True:
  795. # Great! verify that those warps link back to us!
  796. possible = [ x for x in possible if x in self.game.gamedata.warps and self.this_sector in self.game.gamedata.warps[x]]
  797. if len(possible) == 0:
  798. self.state = 0
  799. self.queue_game.put(self.r + self.nl + "I don't see any ports in [{0}].".format(self.warps) + self.nl)
  800. self.deactivate()
  801. return
  802. possible = [ x for x in possible if not GameData.port_burnt(self.game.gamedata.ports[x]) ]
  803. log.debug("Possible: {0}".format(possible))
  804. if len(possible) == 0:
  805. self.state = 0
  806. self.queue_game.put(self.r + self.nl + "I don't see any unburnt ports in [{0}].".format(self.warps) + self.nl)
  807. self.deactivate()
  808. return
  809. possible = [ x for x in possible if GameData.port_trading(self.game.gamedata.ports[self.this_sector]['port'], self.game.gamedata.ports[x]['port'])]
  810. # sort by best, then by %
  811. start_port = self.game.gamedata.ports[self.this_sector]
  812. if 'port' in start_port:
  813. start_port_port = start_port['port']
  814. start_with = start_port_port[1:]
  815. if start_with in ('BS', 'SB'):
  816. # Ok, good trades may be possible
  817. best = GameData.flip(start_with)
  818. log.info("Sorting the best ({0}) to the top.".format(best))
  819. log.info("{0}".format(possible))
  820. dec = [ [self.game.gamedata.ports[p].get('port', '---')[1:] == best, p] for p in possible]
  821. dec = sorted(dec, reverse=True)
  822. possible = [x[1] for x in dec]
  823. log.info("{0}".format(possible))
  824. self.possible = possible
  825. if len(possible) == 0:
  826. self.state = 0
  827. self.queue_game.put(self.r + self.nl + "I don't see any possible port trades in [{0}].".format(self.warps) + self.nl)
  828. self.deactivate()
  829. return
  830. elif len(possible) == 1:
  831. # Ok! there's only one!
  832. self.sector2 = possible[0]
  833. # Display possible ports:
  834. # spos = [ str(x) for x in possible]
  835. # self.queue_game.put(self.r + self.nl + self.nl.join(spos) + self.nl)
  836. # At state 3, we only get a prompt.
  837. return
  838. else:
  839. self.state = 0
  840. log.warn("We don't have any portdata!")
  841. self.queue_game.put(self.r + self.nl + "I have no portdata. Please run CIM Port Report." + self.nl)
  842. self.deactivate()
  843. return
  844. elif self.state == 5:
  845. if "-----" in line:
  846. self.state = 6
  847. elif self.state == 6:
  848. if "We are buying up to" in line:
  849. log.info("buying up to -- so sell all")
  850. # Sell
  851. self.state = 7
  852. self.queue_player.put("\r")
  853. self.sell_percent = 100 + self.percent
  854. if "We are selling up to" in line:
  855. log.info("selling up to -- state 8 / set percent")
  856. # Buy
  857. self.state = 8
  858. self.sell_percent = 100 - self.percent
  859. if line.startswith('Fuel Ore') or line.startswith('Organics') or line.startswith('Equipment'):
  860. work = line.replace('Fuel Ore', 'Fuel').replace('%', '')
  861. parts = re.split(r"\s+", work)
  862. # log.debug(parts)
  863. # Equipment, Selling xxx x% xxx
  864. if parts[-1] != '0' and parts[2] != '0' and parts[1] != 'Buying':
  865. log.warn("We have a problem -- they aren't buying what we have in stock!")
  866. stuff = line[0] # F O or E.
  867. if self.game.gamedata.port_buying(self.other_sector, stuff):
  868. log.info("fixable")
  869. self.fixable = True
  870. if int(parts[3]) < self.stop:
  871. log.info("Port is burnt! % < {0} (end_trans)".format(self.stop))
  872. self.end_trans = True
  873. if "You don't have anything they want" in line:
  874. log.warn("Don't have anything they want.")
  875. # Neither! DRAT!
  876. if not self.fixable:
  877. self.state = 99
  878. return
  879. if "We're not interested." in line:
  880. log.warn("Try, try again. :(")
  881. self.state = 5
  882. self.trade()
  883. elif self.state == 7:
  884. # Haggle Sell
  885. if "We'll buy them for" in line or "Our final offer" in line:
  886. if "Our final offer" in line:
  887. self.sell_percent -= 1
  888. parts = line.replace(',', '').split()
  889. start_price = int(parts[4])
  890. price = start_price * self.sell_percent // 100
  891. log.debug("start: {0} % {1} price {2}".format(start_price, self.sell_percent, price))
  892. self.sell_percent -= 1
  893. self.queue_player.put("{0}\r".format(price))
  894. if "We are selling up to" in line:
  895. log.info("selling up to / state 8 / set percent")
  896. # Buy
  897. self.state = 8
  898. self.sell_percent = 100 - self.percent
  899. if "We are buying up to" in line:
  900. log.info("buying up to -- so sell all")
  901. # Sell
  902. self.state = 7
  903. self.queue_player.put("\r")
  904. self.sell_percent = 100 + self.percent
  905. if "We're not interested." in line:
  906. log.info("Not interested. Try, try again. :(")
  907. self.state = 5
  908. self.trade()
  909. if "WHAT?!@!? you must be crazy!" in line:
  910. log.warn("fix offer")
  911. self.fix_offer = 1
  912. if "So, you think I'm as stupid as you look?" in line:
  913. log.warn("fix offer")
  914. self.fix_offer = 1
  915. if "Quit playing around, you're wasting my time!" in line:
  916. log.warn("fix offer")
  917. self.fix_offer = 1
  918. elif self.state == 8:
  919. # Haggle Buy
  920. if "We'll sell them for" in line or "Our final offer" in line:
  921. if "Our final offer" in line:
  922. self.sell_percent += 1
  923. parts = line.replace(',', '').split()
  924. start_price = int(parts[4])
  925. price = start_price * self.sell_percent // 100
  926. log.debug("start: {0} % {1} price {2}".format(start_price, self.sell_percent, price))
  927. self.sell_percent += 1
  928. self.queue_player.put("{0}\r".format(price))
  929. if "We're not interested." in line:
  930. log.info("Not interested. Try, try again. :(")
  931. self.state = 5
  932. self.trade()
  933. elif self.state == 10:
  934. if "Sector : " in line:
  935. # Trade
  936. self.state = 5
  937. reactor.callLater(0, self.trade, 0)
  938. # self.trade()
  939. # elif self.state == 3:
  940. # log.debug("At state 3 [{0}]".format(line))
  941. # self.queue_game.put("At state 3.")
  942. # self.deactivate()
  943. # return
  944. class ScriptExplore(object):
  945. """ Exploration Script
  946. WARNINGS:
  947. We assume the player has lots o turns, or unlimited turns!
  948. """
  949. def __init__(self, game):
  950. self.game = game
  951. self.queue_game = game.queue_game
  952. self.queue_player = game.queue_player
  953. self.observer = game.observer
  954. self.r = Style.RESET_ALL
  955. self.c = merge(Style.BRIGHT + Fore.YELLOW)
  956. self.nl = "\n\r"
  957. # Our Stuff, Not our pants!
  958. self.dense = [] # We did a density, store that info.
  959. self.clear = [] # Warps that we know are clear.
  960. self.highsector = 0 # Selected Sector to move to next!
  961. self.highwarp = 0 # Selected Sector's Warp Count!
  962. self.stacksector = [] # Set of sectors that we have not picked but are unexplored... even though we did a holo!
  963. self.oneMoveSector = False
  964. self.times = 0
  965. self.maxtimes = 0
  966. # Activate
  967. self.prompt = game.buffer
  968. self.save = self.observer.save()
  969. self.observer.connect('player', self.player)
  970. self.observer.connect("prompt", self.game_prompt)
  971. self.observer.connect("game-line", self.game_line)
  972. self.prefer_ports = self.game.gamedata.get_config('Explorer_PrefPorts', 'N').upper()[0] == 'Y'
  973. self.defer = None
  974. self.send2player(Boxes.alert("Explorer", base="green"))
  975. # How many times we going to go today?
  976. ask = PlayerInput(self.game)
  977. def settimes(*_):
  978. times = ask.keep['times'].strip()
  979. log.debug("settimes got '{0}'".format(times))
  980. if times == '':
  981. self.deactivate()
  982. else:
  983. times = int(times)
  984. self.times = times
  985. self.maxtimes = times
  986. self.send2game("D")
  987. log.debug("times: {0} maxtimes: {0}".format(self.times))
  988. self.state = 1
  989. d = ask.prompt("How many sectors would you like to explorer?", 5, name="times", digits=True)
  990. #d.addCallback(ask.output)
  991. #d.addCallback(lambda ignore: self.settimes(ask.keep))
  992. d.addCallback(settimes)
  993. def whenDone(self):
  994. self.defer = defer.Deferred()
  995. # Call this to chain something after we exit.
  996. return self.defer
  997. def deactivate(self, andExit=False):
  998. self.state = 0
  999. log.debug("ScriptExplore.deactivate()")
  1000. assert(not self.save is None)
  1001. self.observer.load(self.save)
  1002. self.save = None
  1003. if self.defer:
  1004. if andExit:
  1005. self.defer.callback({'exit':True})
  1006. else:
  1007. self.defer.callback('done')
  1008. self.defer = None
  1009. def player(self, chunk: bytes):
  1010. # If we receive anything -- ABORT!
  1011. self.deactivate(True)
  1012. def send2game(self, txt):
  1013. log.debug("ScriptExplore.send2game({0})".format(txt))
  1014. self.queue_player.put(txt)
  1015. def send2player(self, txt):
  1016. log.debug("ScriptExplore.send2player({0})".format(txt))
  1017. self.queue_game.put(txt)
  1018. def resetStuff(self):
  1019. self.dense = []
  1020. self.clear = []
  1021. self.highwarp = 0
  1022. self.highsector = 0
  1023. log.debug("ScriptExplore.resetStuff()")
  1024. def dead_end(self):
  1025. """ We've reached a dead end.
  1026. Either pop a new location to travel to, or give it up.
  1027. """
  1028. self.send2player(self.nl + Boxes.alert("** DEAD END **", base="blue"))
  1029. if self.stacksector:
  1030. # Ok, there's somewhere to go ...
  1031. self.highsector = self.stacksector.pop()
  1032. # travel state
  1033. self.state = 10
  1034. self.send2game("{0}\r".format(self.highsector))
  1035. else:
  1036. self.send2player(self.nl + Boxes.alert("I've run out of places to look ({0}/{1}).".format(self.maxtimes - self.times, self.maxtimes)))
  1037. self.deactivate(True)
  1038. def game_over(self, msg):
  1039. self.send2player(self.nl + Boxes.alert("STOP: {0} ({1}/{2}).".format(msg, self.maxtimes - self.times, self.maxtimes)))
  1040. self.deactivate()
  1041. def game_prompt(self, prompt: str):
  1042. log.debug("{0} : {1}".format(self.state, prompt))
  1043. if self.state == 2:
  1044. if "Select (H)olo Scan or (D)ensity Scan or (Q)uit" in prompt:
  1045. self.send2game("D")
  1046. # self.state += 1
  1047. elif self.state == 5:
  1048. log.debug("dense is {0} sectors big".format(len(self.dense)))
  1049. self.state += 1
  1050. self.send2game("SH")
  1051. elif self.state == 10:
  1052. if prompt.startswith('Do you want to engage the TransWarp drive? '):
  1053. self.send2game("N")
  1054. elif self.state == 12:
  1055. # Looking for "Engage the Autopilot?"
  1056. if prompt.startswith('Engage the Autopilot? (Y/N/Single step/Express) [Y]'):
  1057. self.send2game("S")
  1058. self.travel_path.pop(0)
  1059. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N]'):
  1060. self.send2game("SD")
  1061. self.state += 1
  1062. # Arriving sector :1691 Autopilot disengaging.
  1063. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1064. log.info("We made it to where we wanted to go!")
  1065. # can't init state 1, because we're at a prompt, so...
  1066. self.send2game("S")
  1067. self.state = 2
  1068. return
  1069. elif self.state == 15:
  1070. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N]'):
  1071. self.send2game("N")
  1072. self.travel_path.pop(0)
  1073. self.state = 12
  1074. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1075. log.info("We made it to where we wanted to go!")
  1076. # can't init state 1, because we're at a prompt, so...
  1077. self.send2game("S")
  1078. self.state = 2
  1079. return
  1080. elif self.state == 20:
  1081. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N]'):
  1082. # Stop in this sector / Yes!
  1083. self.send2game("Y")
  1084. self.state = 1
  1085. # this should re-trigger a scan
  1086. def game_line(self, line: str):
  1087. log.debug("{0} | {1}".format(self.state, line))
  1088. #if "Mine Control" in line: # If we don't have a Holo-Scanner and we attempted to do a Holo-scan, abort
  1089. # self.deactivate()
  1090. if line.startswith("You don't have enough turns left."):
  1091. self.send2player(self.nl + Boxes.alert("You're out of turns!"))
  1092. self.deactivate(True)
  1093. return
  1094. if self.state == 1:
  1095. if line.startswith('You have ') and 'turns left.' in line:
  1096. # Ok, you're in trouble!
  1097. self.send2player(self.nl + Boxes.alert("You're running low on turns!"))
  1098. self.deactivate(True)
  1099. return
  1100. self.send2game("S")
  1101. self.state += 1
  1102. elif self.state == 2:
  1103. if "Relative Density Scan" in line:
  1104. self.state = 3
  1105. elif "You don't have a long range scanner." in line:
  1106. log.warn("FATAL: No Long Range Scanner Installed!")
  1107. self.send2player(Boxes.alert("You need a Long Range Scanner!"))
  1108. self.deactivate(True)
  1109. return
  1110. # elif "Long Range Scan" in line:
  1111. # self.state += 1
  1112. elif self.state == 3:
  1113. # Get the Density Data!
  1114. if line.startswith("Sector"):
  1115. new_sector = '(' in line
  1116. work = line.replace(':', '').replace(')', '').replace('(', '').replace('%', '').replace('==>', '')
  1117. work = re.split(r"\s+", work)
  1118. log.debug(work)
  1119. # 'Sector', '8192', '0', 'Warps', '4', 'NavHaz', '0', 'Anom', 'No'
  1120. # 'Sector', '(', '8192)', '0', 'Warps', '4', 'NavHaz', '0', 'Anom', 'No'
  1121. # New Sector?
  1122. if new_sector:
  1123. # Switch Anom into bool state
  1124. # if(work[8] == 'No'):
  1125. # temp = False
  1126. # else:
  1127. # temp = True
  1128. #self.dense.append( {'sector': int(work[1]), 'density': int(work[2]), 'warps': int(work[4]), 'navhaz': int(work[6]), 'anom': temp} )
  1129. self.dense.append( {'sector': int(work[1]), 'density': int(work[2]), 'warps': int(work[4]), 'navhaz': int(work[6]), 'anom': work[8] == 'Yes'} )
  1130. log.debug(self.dense)
  1131. # {'sector': 8192, 'density': 0, 'warps': 4, 'navhaz': 0, 'anom': False}
  1132. elif line == "":
  1133. self.state += 1
  1134. # yeah, this would be better in the above line...
  1135. # leaving it for now.
  1136. # Which is why I broke the elif chain. ...
  1137. if self.state == 4:
  1138. # Begin Processing our data we got from density scan and find highest warp count in what sectors
  1139. # Remove sectors with one warp
  1140. log.debug("state 4: {0}".format(self.dense))
  1141. # Do we have a new place to go? (That is also worth going to)
  1142. if not self.dense: # Dense contains no new sectors, abort
  1143. log.info("No New Sectors Found!")
  1144. self.dead_end()
  1145. return
  1146. # Is the sector safe to go into?
  1147. # self.clear = [ x['sector'] for x in self.dense if not x['anom'] and not x['navhaz'] and x['density'] in (0,1,100,101) and x['warps'] > 1 ]
  1148. self.clear = [ x for x in self.dense if not x['anom'] and not x['navhaz'] and x['density'] in (0,1,100,101) ]
  1149. if self.clear: # We have sector(s) we can move to!
  1150. log.debug("Clear Sectors: {0}".format(len(self.clear)))
  1151. # This was state 5 but why can't we reduce number of states? ( Yeah let's kick California and New York out of the US, oh wrong states :P )
  1152. # Sort to find greatest warp count
  1153. if self.prefer_ports:
  1154. _, self.highwarp, self.highsector = max( (x['density'], x['warps'], x['sector']) for x in self.clear)
  1155. else:
  1156. self.highwarp, self.highsector = max( (x['warps'], x['sector']) for x in self.clear)
  1157. log.info("Sector: {0:5d} Warps: {1}".format(self.highsector, self.highwarp))
  1158. self.state += 1
  1159. else:
  1160. log.warn("No (safe) sectors to move to!")
  1161. # Let's try this?!
  1162. # self.dead_end() # NO!
  1163. self.game_over("No SAFE moves.")
  1164. # Another NOP state. This also could be merged into above.
  1165. # break the elif chain.
  1166. if self.state == 5:
  1167. # Add the dense scan of unknown sectors onto the stack of sectors, only save the ones we think are clear... for now.
  1168. for c in self.clear:
  1169. sector = c['sector']
  1170. if sector != self.highsector:
  1171. if sector not in self.stacksector:
  1172. self.stacksector.append(sector)
  1173. # Or simply not add it in the first place ...
  1174. # Remove the sector we are just about to go to, we use discard so if the sector does not exist we don't throw a error!
  1175. # self.stacksector.discard(self.highsector)
  1176. # Ok, we need to decide to stop exploring -- before we
  1177. # issue the sector move! :P
  1178. #
  1179. # Warning! Yes we can and will eat all the turns! :P
  1180. if self.times == 0:
  1181. self.send2player(Boxes.alert("Completed {0}".format(self.maxtimes), base="green"))
  1182. log.info("Completed {0}".format(self.maxtimes))
  1183. self.deactivate()
  1184. return
  1185. self.times -= 1
  1186. # Ok we know the sector we want to go to now let's move it!
  1187. self.send2game("m{0}\r".format(self.highsector))
  1188. if self.highsector in self.stacksector:
  1189. log.info("Removing {0} from stacksector list.".format(self.highsector))
  1190. self.stacksector.remove(self.highsector)
  1191. # Reset Variables for fresh data
  1192. self.resetStuff()
  1193. self.state = 1
  1194. elif self.state == 10:
  1195. if line.startswith("You are already in that sector!"):
  1196. log.info("Already here. (Whoops!)")
  1197. self.state = 1
  1198. return
  1199. if line.startswith("Sector : {0}".format(self.highsector)):
  1200. log.info("We're here!")
  1201. # Ok, we're already there! no autopilot needed!
  1202. self.state = 1
  1203. return
  1204. # Warping
  1205. self.go_on = True
  1206. if line.startswith('The shortest path ('):
  1207. # Ok, we've got a path.
  1208. self.state += 1
  1209. self.travel_path = []
  1210. elif self.state == 11:
  1211. if line == '':
  1212. # The end of the (possibly) multiline warp.
  1213. self.state += 1
  1214. self.travel_path.pop(0) # First sector is one we're in.
  1215. self.stophere = False
  1216. self.go_on = True
  1217. else:
  1218. self.travel_path.extend(line.replace('(', '').replace(')', '').split(' > ') )
  1219. log.debug("Travel path: {0}".format(self.travel_path))
  1220. elif self.state == 12:
  1221. # Arriving sector :1691 Autopilot disengaging.
  1222. if 'Autopilot disengaging.' in line:
  1223. log.info("We made it to where we wanted to go!")
  1224. self.state = 1
  1225. return
  1226. elif self.state == 13:
  1227. if 'Relative Density Scan' in line:
  1228. self.state += 1
  1229. elif self.state == 14:
  1230. if line == "":
  1231. log.debug("PATH: {0}".format(self.travel_path))
  1232. # end of the scan, decision time
  1233. if self.stophere:
  1234. log.info("STOPHERE")
  1235. # Ok, let's stop here!
  1236. # Re-save the sector we were trying to get to. (we didn't make it there)
  1237. if self.highsector not in self.stacksector:
  1238. self.stacksector.append(self.highsector)
  1239. self.state = 20
  1240. else:
  1241. if self.go_on:
  1242. log.info("GO ON")
  1243. # Ok, carry on!
  1244. self.state = 15
  1245. else:
  1246. log.warn("Our way is blocked...")
  1247. if self.highsector not in self.stacksector:
  1248. self.stacksector.append(self.highsector)
  1249. self.state = 20
  1250. else:
  1251. if line.strip('-') != '':
  1252. work = line.replace(' :', '').replace('%', '').replace(')', '').replace('==>', '')
  1253. # Does this contain something new? unseen?
  1254. stophere = '(' in work
  1255. work = work.replace('(','')
  1256. #Sector XXXX DENS Warps N NavHaz P Anom YN
  1257. parts = re.split(r'\s+', work)
  1258. # Don't bother stopping if there's only one warp
  1259. # YES! Stop, even if there is just one warp!
  1260. # if stophere and parts[4] == '1':
  1261. # stophere = False
  1262. if stophere:
  1263. self.stophere = True
  1264. next_stop = self.travel_path[0]
  1265. log.debug("next_stop {0} from {1}".format(next_stop, self.travel_path))
  1266. log.debug("parts: {0}".format(parts))
  1267. if parts[1] == next_stop:
  1268. log.info("next_stop {0} found...".format(next_stop))
  1269. # Ok, this is our next stop. Is it safe to travel to?
  1270. if parts[2] not in ('100', '0', '1', '101'):
  1271. # Ok, it's not safe to go on.
  1272. self.go_on = False
  1273. # Check the rest navhav and anom ...
  1274. class ScriptSpace(object):
  1275. """ Space Exploration script.
  1276. Send "SD", verify paths are clear.
  1277. Find nearest unknown. Sector + CR.
  1278. Save "shortest path from to" information.
  1279. At 'Engage the Autopilot', Send "S" (Single Step)
  1280. At '[Stop in this sector', Send "SD", verify path is clear.
  1281. Send "SH" (glean sector/port information along the way.)
  1282. Send "N" (Next)!
  1283. Send "SD" / Verify clear.
  1284. Send "SH"
  1285. Repeat for Next closest.
  1286. """
  1287. def __init__(self, game):
  1288. self.game = game
  1289. self.queue_game = game.queue_game
  1290. self.queue_player = game.queue_player
  1291. self.observer = game.observer
  1292. self.r = Style.RESET_ALL
  1293. self.nl = "\n\r"
  1294. self.this_sector = None # Starting sector
  1295. self.target_sector = None # Sector going to
  1296. self.path = []
  1297. self.times_left = 0 # How many times to look for target
  1298. self.density = dict() # Results of density scan. (Just the numbers)
  1299. # Activate
  1300. self.prompt = game.buffer
  1301. self.save = self.observer.save()
  1302. self.observer.connect('player', self.player)
  1303. self.observer.connect("prompt", self.game_prompt)
  1304. self.observer.connect("game-line", self.game_line)
  1305. self.defer = None
  1306. self.queue_game.put(
  1307. self.nl + "Bugz (like space), is big." + self.r + self.nl
  1308. )
  1309. self.state = 1
  1310. self.queue_player.put("SD")
  1311. # Get current density scan + also get the current sector.
  1312. # [Command [TL=00:00:00]:[XXXX] (?=Help)? : D]
  1313. def whenDone(self):
  1314. self.defer = defer.Deferred()
  1315. # Call this to chain something after we exit.
  1316. return self.defer
  1317. def deactivate(self, andExit=False):
  1318. self.state = 0
  1319. log.debug("ScriptPort.deactivate ({0})".format(self.times_left))
  1320. assert(not self.save is None)
  1321. self.observer.load(self.save)
  1322. self.save = None
  1323. if self.defer:
  1324. if andExit:
  1325. self.defer.callback({'exit':True})
  1326. else:
  1327. self.defer.callback('done')
  1328. self.defer = None
  1329. def player(self, chunk: bytes):
  1330. # If we receive anything -- ABORT!
  1331. self.deactivate(True)
  1332. def unknown_search(self, starting_sector):
  1333. seen = set()
  1334. possible = set()
  1335. possible.add(int(starting_sector))
  1336. done = False
  1337. while not done:
  1338. next_possible = set()
  1339. for s in possible:
  1340. p = self.game.gamedata.get_warps(s)
  1341. if p is not None:
  1342. for pos in p:
  1343. if pos not in seen:
  1344. next_possible.add(pos)
  1345. else:
  1346. log.debug("unknown found: {0}".format(s))
  1347. self.unknown = s
  1348. done = True
  1349. break
  1350. seen.add(s)
  1351. if self.unknown is None:
  1352. log.debug("possible: {0}".format(next_possible))
  1353. possible = next_possible
  1354. yield
  1355. def find_unknown(self, starting_sector):
  1356. log.debug("find_unknown( {0})".format(starting_sector))
  1357. d = defer.Deferred()
  1358. # Process things
  1359. self.unknown = None
  1360. c = coiterate(self.unknown_search(starting_sector))
  1361. c.addCallback(lambda unknown: d.callback(self.unknown))
  1362. return d
  1363. def show_unknown(self, sector):
  1364. if sector is None:
  1365. self.deactivate()
  1366. return
  1367. log.debug("Travel to {0}...".format(sector))
  1368. self.queue_player.put("{0}\r".format(sector))
  1369. def game_prompt(self, prompt: str):
  1370. log.debug("{0} : {1}".format(self.state, prompt))
  1371. if self.state == 3:
  1372. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1373. # this_sector code isn't working -- so! Get sector from prompt
  1374. self.state = 4
  1375. _, _, sector = prompt.partition(']:[')
  1376. sector, _, _ = sector.partition(']')
  1377. self.this_sector = int(sector)
  1378. # Ok, we're done with Density Scan, and we're back at the command prompt
  1379. log.debug("Go find the nearest unknown...")
  1380. d = self.find_unknown(sector)
  1381. d.addCallback(self.show_unknown)
  1382. elif self.state == 6:
  1383. # Engage the autopilot?
  1384. if prompt.startswith('Engage the Autopilot? (Y/N/Single step/Express) [Y]'):
  1385. self.state = 7
  1386. sector = self.path.pop(0)
  1387. if sector in self.density:
  1388. if self.density[sector] in (0, 100):
  1389. # Ok, looks safe!
  1390. self.queue_player.put("S")
  1391. self.this_sector = sector
  1392. else:
  1393. log.warn("FATAL: Density for {0} is {1}".format(sector, self.density[sector]))
  1394. self.deactivate(True)
  1395. return
  1396. else:
  1397. log.error("{0} not in density scan? (how's that possible?)".format(sector))
  1398. self.deactivate(True)
  1399. return
  1400. elif self.state == 7:
  1401. # Ok, we're in a new sector (single stepping through space)
  1402. # update density scan
  1403. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1404. self.queue_player.put("SD")
  1405. self.state = 8
  1406. elif self.state == 10:
  1407. # Because we're here
  1408. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1409. self.queue_player.put("SH")
  1410. self.state = 11
  1411. elif self.state == 11:
  1412. if prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1413. # Ok, is the density scan clear?
  1414. sector = self.path.pop(0)
  1415. if sector in self.density:
  1416. if self.density[sector] in (0, 100):
  1417. # Ok, looks safe
  1418. self.queue_player.put("N")
  1419. self.state = 7
  1420. self.this_sector = sector
  1421. else:
  1422. log.warn("FATAL: Density for {0} is {1}".format(sector, self.density[sector]))
  1423. self.deactivate()
  1424. return
  1425. else:
  1426. log.error("{0} not in density scane? (how's that possible...)".format(sector))
  1427. self.deactivate()
  1428. return
  1429. def next_unknown(self, sector):
  1430. log.info("Unknown is : {0}".format(sector))
  1431. self.deactivate()
  1432. def game_line(self, line: str):
  1433. log.debug("line {0} : {1}".format(self.state, line))
  1434. if line.startswith('Sector : '):
  1435. work = line.strip()
  1436. parts = re.split(r"\s+", work)
  1437. self.this_sector = int(parts[2])
  1438. log.debug("game_line sector {0}".format(self.this_sector))
  1439. elif line.startswith("Command [TL=]"):
  1440. # Ok, get the current sector from this
  1441. _, _, sector = line.partition("]:[")
  1442. sector, _, _ = sector.partition("]")
  1443. self.this_sector = int(sector)
  1444. log.debug("current sector: {0}".format(self.this_sector))
  1445. elif line.startswith('Warps to Sector(s) :'):
  1446. # Warps to Sector(s) : 5468
  1447. _, _, work = line.partition(':')
  1448. work = work.strip().replace('(', '').replace(')', '').replace(' - ', ' ')
  1449. parts = [ int(x) for x in work.split(' ')]
  1450. self.path = list(parts)
  1451. if self.state in (1, 8):
  1452. if 'Relative Density Scan' in line:
  1453. # Start Density Scan
  1454. self.state += 1
  1455. self.density = {}
  1456. elif self.state in (2, 9):
  1457. if line == '':
  1458. # End of Density Scan
  1459. self.state += 1
  1460. log.debug("Density: {0}".format(self.density))
  1461. # self.deactivate()
  1462. elif line.startswith('Sector'):
  1463. # Parse Density Scan values
  1464. work = line.replace('(', '').replace(')', '').replace(':', '').replace('%', '').replace(',', '')
  1465. parts = re.split(r'\s+', work)
  1466. log.debug("Sector {0}".format(parts))
  1467. sector = int(parts[1])
  1468. self.density[sector] = int(parts[3])
  1469. if parts[7] != '0':
  1470. log.warn("NavHaz {0} : {1}".format(parts[7], work))
  1471. self.density[sector] += 99
  1472. if parts[9] != 'No':
  1473. log.warn("Anom {0} : {1}".format(parts[9], work))
  1474. self.density[sector] += 990
  1475. elif self.state == 4:
  1476. # Looking for shortest path message / warp info
  1477. # Or possibly, "We're here!"
  1478. if line.startswith('Sector :') and str(self.unknown) in line:
  1479. # Ok, I'd guess that we're already there!
  1480. # Try it again!
  1481. self.queue_player.put("SD")
  1482. self.state = 1
  1483. if line.startswith('The shortest path'):
  1484. self.state = 5
  1485. elif self.state == 5:
  1486. # This is the warps line
  1487. # Can this be multiple lines?
  1488. if line == "":
  1489. self.state = 6
  1490. else:
  1491. work = line.replace("(", "").replace(")", "").replace(">", "").strip()
  1492. self.path = [int(x) for x in work.split()]
  1493. log.debug("Path: {0}".format(self.path))
  1494. # Verify
  1495. current = self.path.pop(0)
  1496. if current != self.this_sector:
  1497. log.warn("Failed: {0} != {1}".format(current, self.this_sector))
  1498. self.deactivate()
  1499. return
  1500. elif self.state == 7:
  1501. if self.unknown == self.this_sector:
  1502. # We have arrived!
  1503. log.info("We're here!")
  1504. self.deactivate()
  1505. class ScriptTerror(object):
  1506. """ Terror script.
  1507. This uses the Port Trading script.
  1508. Basically, we look for the next best port trading pair.
  1509. Move to it, fire off the Port Trading script.
  1510. Repeat until our loop is done, or there's no more
  1511. pairs.
  1512. """
  1513. def __init__(self, game, proxy, count):
  1514. self.game = game
  1515. self.queue_game = game.queue_game
  1516. self.queue_player = game.queue_player
  1517. self.proxy = proxy
  1518. self.count = count
  1519. self.observer = game.observer
  1520. self.r = Style.RESET_ALL
  1521. self.nl = "\n\r"
  1522. self.target_sector = None
  1523. # Activate
  1524. self.prompt = game.buffer
  1525. self.save = self.observer.save()
  1526. self.observer.connect('player', self.player)
  1527. self.observer.connect("prompt", self.game_prompt)
  1528. self.observer.connect("game-line", self.game_line)
  1529. self.state = 0
  1530. self.defer = None
  1531. # Verify that they have configured auto-port trading.
  1532. # Otherwise, this doesn't work!
  1533. usefirst = self.game.gamedata.get_config('Trade_UseFirst')
  1534. if usefirst is None:
  1535. usefirst = '?'
  1536. if usefirst.upper()[0] != 'Y':
  1537. self.queue_game.put(Boxes.alert("Sorry! You need to config Trade_UseFirst=Y", base="red"))
  1538. self.deactivate()
  1539. return
  1540. turns = self.game.gamedata.get_config('Trade_Turns')
  1541. if turns is None:
  1542. turns = '0'
  1543. try:
  1544. t = int(turns)
  1545. except ValueError:
  1546. self.queue_game.put(Boxes.alert("Sorry! You need to config Trade_Turns", base="red"))
  1547. self.deactivate()
  1548. return
  1549. if t < 5:
  1550. self.queue_game.put(Boxes.alert("Sorry! You need to config Trade_Turns >", base="red"))
  1551. self.deactivate()
  1552. return
  1553. c = coiterate(self.find_next_good_trade_pair())
  1554. c.addCallback(lambda unknown: self.scary())
  1555. def scary(self):
  1556. if self.target_sector is None:
  1557. self.queue_game.put(Boxes.alert("Sorry! I don't see any ports to trade with.", base="red"))
  1558. self.deactivate()
  1559. else:
  1560. self.state = 1
  1561. self.queue_player.put("{0}\r".format(self.target_sector))
  1562. def find_next_good_trade_pair(self):
  1563. """ Find the next GOOD trade pair sector. """
  1564. show_limit = 90
  1565. # Look for "GOOD" trades
  1566. for sector in sorted(self.game.gamedata.ports.keys()):
  1567. pd = self.game.gamedata.ports[sector]
  1568. if not GameData.port_burnt(pd):
  1569. pc = pd['class']
  1570. # Ok, let's look into it.
  1571. if not sector in self.game.gamedata.warps:
  1572. continue
  1573. warps = self.game.gamedata.warps[sector]
  1574. for w in warps:
  1575. # We can get there, and get back.
  1576. if w in self.game.gamedata.warps and sector in self.game.gamedata.warps[w]:
  1577. # Ok, we can get there -- and get back!
  1578. if w > sector and w in self.game.gamedata.ports and not GameData.port_burnt(self.game.gamedata.ports[w]):
  1579. wd = self.game.gamedata.ports[w]
  1580. wc = wd['class']
  1581. if pc in (1,5) and wc in (2,4):
  1582. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  1583. if data:
  1584. self.target_sector = sector
  1585. return sector
  1586. elif pc in (2,4) and wc in (1,5):
  1587. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  1588. if data:
  1589. self.target_sector = sector
  1590. return sector
  1591. yield
  1592. # Look for OK trades
  1593. for sector in sorted(self.game.gamedata.ports.keys()):
  1594. pd = self.game.gamedata.ports[sector]
  1595. if not GameData.port_burnt(pd):
  1596. pc = pd['class']
  1597. # Ok, let's look into it.
  1598. if not sector in self.game.gamedata.warps:
  1599. continue
  1600. warps = self.game.gamedata.warps[sector]
  1601. for w in warps:
  1602. # We can get there, and get back.
  1603. if w in self.game.gamedata.warps and sector in self.game.gamedata.warps[w]:
  1604. # Ok, we can get there -- and get back!
  1605. if w > sector and w in self.game.gamedata.ports and not GameData.port_burnt(self.game.gamedata.ports[w]):
  1606. wd = self.game.gamedata.ports[w]
  1607. wc = wd['class']
  1608. if GameData.port_trading(pd['port'], self.game.gamedata.ports[w]['port']):
  1609. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  1610. if data:
  1611. self.target_sector = sector
  1612. return sector
  1613. yield
  1614. self.target_sector = None
  1615. def whenDone(self):
  1616. self.defer = defer.Deferred()
  1617. # Call this to chain something after we exit.
  1618. return self.defer
  1619. def deactivate(self, andExit=False):
  1620. self.state = 0
  1621. log.debug("ScriptTerror.deactivate")
  1622. assert(not self.save is None)
  1623. self.observer.load(self.save)
  1624. self.save = None
  1625. if self.defer:
  1626. if andExit:
  1627. self.defer.callback({'exit':True})
  1628. else:
  1629. self.defer.callback('done')
  1630. self.defer = None
  1631. def player(self, chunk: bytes):
  1632. # If we receive anything -- ABORT!
  1633. self.deactivate(True)
  1634. def journey_on(self, *_):
  1635. log.info("journey_on( {0})".format(self.count))
  1636. if self.count > 0:
  1637. self.count -= 1
  1638. c = coiterate(self.find_next_good_trade_pair())
  1639. c.addCallback(lambda unknown: self.scary())
  1640. # self.target_sector = self.proxy.find_next_good_trade_pair() # Sector going to
  1641. # self.state = 1
  1642. # self.queue_player.put("{0}\r".format(self.target_sector))
  1643. else:
  1644. self.deactivate()
  1645. def game_prompt(self, prompt: str):
  1646. log.debug("{0} : {1}".format(self.state, prompt))
  1647. if self.state == 1:
  1648. if prompt.startswith('Do you want to engage the TransWarp drive? '):
  1649. self.queue_player.put("N")
  1650. elif self.state == 2:
  1651. if prompt.startswith('Engage the Autopilot? (Y/N/Single step/Express) [Y]'):
  1652. self.queue_player.put("E")
  1653. elif prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1654. self.queue_player.put("N")
  1655. elif re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1656. # We should be where we wanted to.
  1657. ports = ScriptPort(self.game)
  1658. d = ports.whenDone()
  1659. d.addCallback(self.journey_on)
  1660. d.addErrback(self.journey_on)
  1661. def game_line(self, line: str):
  1662. log.debug("line {0} : {1}".format(self.state, line))
  1663. if self.state == 1:
  1664. if line.startswith('The shortest path ('):
  1665. self.state = 2
  1666. elif line.startswith("You are already in that sector!"):
  1667. # Whoops.
  1668. ports = ScriptPort(self.game)
  1669. d = ports.whenDone()
  1670. d.addCallback(self.journey_on)
  1671. d.addErrback(self.journey_on)
  1672. class PlanetUpScript(object):
  1673. def __init__(self, game):
  1674. self.game = game
  1675. self.queue_game = game.queue_game
  1676. self.queue_player = game.queue_player
  1677. self.observer = game.observer
  1678. # Yes, at this point we would activate
  1679. self.prompt = game.buffer
  1680. self.save = self.observer.save()
  1681. self.nl = "\n\r"
  1682. self.cargo_index = { 'F': 0, 'O': 1, 'E': 2}
  1683. self.index_cargo = ('F', 'O', 'E')
  1684. # I actually don't want the player input, but I'll grab it anyway.
  1685. self.observer.connect("player", self.player)
  1686. self.observer.connect("prompt", self.game_prompt)
  1687. self.observer.connect("game-line", self.game_line)
  1688. # If we want it, it's here.
  1689. self.defer = None
  1690. self.to_player = self.game.to_player
  1691. self.planets = {}
  1692. # Hide what's happening from the player
  1693. self.game.to_player = False
  1694. self.queue_player.put("CYQ") # Computer -> Your Planets -> Quit
  1695. self.state = 1
  1696. # self.warpdata = {}
  1697. self.queue_game.put(Boxes.alert("Let me see what I can see here..."))
  1698. def game_prompt(self, prompt):
  1699. log.info("prompt {0} : {1}".format(self.state, prompt))
  1700. if self.state == 2 and re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1701. self.game.to_player = True
  1702. # For now we output the information, and exit
  1703. # self.queue_game.put("{0}\r\n".format(self.planets))
  1704. # self.queue_game.put(pformat(self.planets).replace("\n", self.nl) + self.nl)
  1705. if len(self.planets) == 0:
  1706. # Ok, this is easy.
  1707. self.queue_game.put(self.nl + Boxes.alert("You don't have any planets? You poor, poor dear.", base="red"))
  1708. self.deactivate()
  1709. return
  1710. # Get current sector from the prompt
  1711. # Command [TL=00:00:00]:[10202] (?=Help)? :
  1712. _, _, part = prompt.partition(']:[')
  1713. sector, _, _ = part.partition(']')
  1714. self.current_sector = int(sector)
  1715. # self.queue_game.put("Current: {0}\r\n".format(sector))
  1716. # Only ONE planet right in front of us? Default to upgrading it!
  1717. # Otherwise, we'll have to ask.
  1718. obvious = [ s for s in self.planets.keys() if self.planets[s]['sector'] == self.current_sector ]
  1719. if len(obvious) == 1:
  1720. # There is one.
  1721. self.planet_number = obvious[0]
  1722. self.planet_sector = self.planets[self.planet_number]['sector']
  1723. self.planet_name = self.planets[self.planet_number]['name']
  1724. self.state = 4
  1725. # begin landing procedure ...
  1726. self.queue_player.put("L")
  1727. return
  1728. # There are either no obvious planets to upgrade,
  1729. # or there are multiple planet choices.
  1730. tc = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  1731. c1 = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  1732. c2 = merge(Style.BRIGHT + Fore.CYAN + Back.BLUE)
  1733. box = Boxes(44, color=tc)
  1734. self.queue_game.put(box.top())
  1735. self.queue_game.put(box.row(tc + "{0:3} {1:6} {2:33}".format(" # ", "Sector", "Planet Name")))
  1736. self.queue_game.put(box.middle())
  1737. def planet_output(number, sector, name):
  1738. row = "{0}{1:^3} {2:6} {3}{4:33}".format(c1, number, sector, c2, name)
  1739. self.queue_game.put(box.row(row))
  1740. for s in sorted(self.planets.keys()):
  1741. planet_output(s, self.planets[s]['sector'], self.planets[s]['name'])
  1742. self.queue_game.put(box.bottom())
  1743. ask = PlayerInput(self.game)
  1744. d = ask.prompt("Choose a planet", 3, name="planet", digits=True)
  1745. d.addCallback(self.planet_chosen)
  1746. elif self.state == 3:
  1747. if prompt.startswith('Do you want to engage the TransWarp drive? '):
  1748. self.queue_player.put("N")
  1749. elif prompt.startswith('Engage the Autopilot? (Y/N/Single step/Express) [Y]'):
  1750. self.queue_player.put("E")
  1751. elif prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1752. self.queue_player.put("N")
  1753. elif re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1754. # We're here!
  1755. self.state = 4
  1756. self.queue_player.put("L")
  1757. elif self.state == 4:
  1758. # If you have a planet scanner, or if there's more then one planet.
  1759. if prompt.startswith('Land on which planet <Q to abort> ?'):
  1760. self.queue_player.put("{0}\r".format(self.planet_number))
  1761. if prompt.startswith('Planet command (?=help) [D] '):
  1762. # self.queue_game.put(self.nl + "{0} : {1}".format( self.colonists, self.cargo) + self.nl)
  1763. self.state = 5
  1764. self.queue_player.put("C")
  1765. elif self.state == 5:
  1766. if re.match(r'Do you wish to construct .+\?', prompt):
  1767. # 'Do you wish to construct a Combat Control Computer?'
  1768. # Have we met all the needs? If so, do it. ;)
  1769. # If not, xfer the cargo on the ship to the planet and get moving!
  1770. ready = True
  1771. for i in self.index_cargo: # ['F', 'O', 'E']:
  1772. if self.need[i] > self.cargo[i]:
  1773. ready = False
  1774. log.info("Need: {0}".format(i))
  1775. # break
  1776. if self.need['C'] > self.colonists:
  1777. log.info("Need: people")
  1778. ready = False
  1779. # self.queue_game.put(self.nl + "{0}".format(self.need))
  1780. if ready:
  1781. self.queue_game.put(self.nl + Boxes.alert("Party Planet Pants On!"))
  1782. self.queue_player.put('YQ')
  1783. self.deactivate()
  1784. return
  1785. if 'construct one' in prompt:
  1786. # No, but start moving things around to build one.
  1787. self.queue_player.put("N")
  1788. else:
  1789. # No, and quit the Citadel menu.
  1790. self.queue_player.put("NQ")
  1791. # Xfer cargo, and get ready to travel...
  1792. elif prompt.startswith('Citadel command (?=help)'):
  1793. self.queue_player.put('U')
  1794. if prompt.startswith('Planet command (?=help) [D] '):
  1795. # self.queue_game.put(pformat(self.ship_cargo).replace("\n", self.nl) + self.nl)
  1796. # self.queue_game.put(pformat(self.cargo).replace("\n", self.nl) + self.nl)
  1797. for idx, c in enumerate(self.index_cargo): # ('F', 'O', 'E')):
  1798. if self.ship_cargo[c] > 0:
  1799. # Transfer Cargo, (No display), Leave [1,2, or 3], \r = All of it.
  1800. self.queue_player.put("TNL{0}\r".format(idx + 1))
  1801. self.cargo[c] += self.ship_cargo[c]
  1802. self.ship_cargo[c] = 0
  1803. return
  1804. break
  1805. self.queue_player.put("Q")
  1806. self.state = 6
  1807. # self.queue_game.put(pformat(self.ship_cargo).replace("\n", self.nl) + self.nl)
  1808. # self.queue_game.put(pformat(self.cargo).replace("\n", self.nl) + self.nl)
  1809. # self.deactivate()
  1810. elif self.state == 6:
  1811. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1812. # Ok, what do we need and where do we get it?
  1813. if self.need['C'] > self.colonists:
  1814. # NavPoint, T, Express
  1815. self.fetch = 'C'
  1816. self.queue_player.put("NTE")
  1817. self.state = 7
  1818. else:
  1819. for i in ('F','O','E'):
  1820. if self.need[i] > self.cargo[i]:
  1821. self.fetch = i
  1822. # TODO: Make this a config setting.
  1823. place = self.game.gamedata.find_nearest_selling(self.planet_sector, i, 400)
  1824. if place == 0:
  1825. self.queue_game.put(self.nl + Boxes.alert("Find Nearest Failed!"))
  1826. self.deactivate()
  1827. return
  1828. self.queue_player.put("{0}\rE".format(place))
  1829. self.state = 7
  1830. return
  1831. # Ok, upgrade time!
  1832. self.state = 4
  1833. self.queue_player.put("L")
  1834. # self.queue_game.put("No, not yet!" + self.nl)
  1835. # self.deactivate()
  1836. # for i in ['F', 'O', 'E']:
  1837. # if self.need[i] > self.cargo[i]:
  1838. # ready = False
  1839. # self.queue_game.put( "Need: {0}".format(i) + self.nl)
  1840. elif self.state == 7:
  1841. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1842. if self.fetch == 'C':
  1843. # Colonist
  1844. # [How many groups of Colonists do you want to take ([125] empty holds) ? ]
  1845. self.queue_player.put("L") # "LT\r")
  1846. elif self.fetch in self.index_cargo: # ('F', 'O', 'E'):
  1847. # Port, Trade
  1848. self.queue_player.put("pt")
  1849. elif re.match(r"How many holds of .+ do you want to sell", prompt):
  1850. # This shouldn't occur...
  1851. self.queue_game.put("OH NOSE!" + self.nl)
  1852. self.deactivate()
  1853. return
  1854. elif prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1855. self.queue_player.put("N")
  1856. elif re.match(r"How many holds of .+ do you want to buy \[\d+\]\? ", prompt):
  1857. if prompt.startswith('How many holds of ' + self.fetch):
  1858. _, _, holds = prompt.partition('[')
  1859. holds, _, _ = holds.partition(']')
  1860. self.fetch_amount = int(holds)
  1861. log.info("Buying {0} of {1}".format(holds, self.fetch))
  1862. self.queue_player.put("\r\r")
  1863. self.state = 8
  1864. else:
  1865. # We don't want to buy this one. Skip it!
  1866. self.queue_player.put("0\r")
  1867. elif prompt.startswith('Land on which planet <Q to abort> ?'):
  1868. if self.fetch == 'C':
  1869. self.queue_player.put("1\r")
  1870. elif prompt.startswith('Do you wish to (L)eave or (T)ake Colonists? [T] (Q to leave)'):
  1871. if self.fetch == 'C':
  1872. self.queue_player.put("T\r")
  1873. self.state = 8
  1874. elif self.state == 8:
  1875. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1876. # Ok, return to the planet...
  1877. self.queue_player.put("{0}\rE".format(self.planet_sector))
  1878. self.state = 9
  1879. elif self.state == 9:
  1880. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  1881. # land
  1882. self.queue_player.put('L')
  1883. elif prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ? '):
  1884. self.queue_player.put("N")
  1885. elif prompt.startswith('Land on which planet <Q to abort> ?'):
  1886. self.queue_player.put("{0}\r".format(self.planet_number))
  1887. elif prompt.startswith('Stop in this sector (Y,N,E,I,R,S,D,P,?) (?=Help) [N] ?'):
  1888. self.queue_player.put("N")
  1889. elif prompt.startswith("Planet command (?=help) [D]"):
  1890. if self.fetch == 'C':
  1891. # Colonist / No display Planet / Leave
  1892. self.queue_player.put("SNL")
  1893. elif self.fetch in self.index_cargo: # ('F', 'O', 'E'):
  1894. # Cargo, No display / Leave
  1895. self.queue_player.put("TNL")
  1896. elif prompt.startswith('(1)Ore, (2)Org or (3)Equipment ?'):
  1897. self.queue_player.put("{0}\r\rQ".format(self.cargo_index[self.fetch] + 1))
  1898. self.cargo[self.fetch] += self.fetch_amount
  1899. self.state = 6
  1900. elif prompt.startswith('(1)Ore, (2)Org or (3)Equipment Production?'):
  1901. log.info("place_people: {0}".format(self.place_people))
  1902. safe_places = [ k for k in self.place_people.keys() if self.place_people[k] ]
  1903. if len(safe_places) == 0:
  1904. # Ok, (GREAT) Class "U" Vaporous/Gaseus >:(
  1905. safe_places = ['F', 'O', 'E']
  1906. # TO FIX: Use self.place_people to decide.
  1907. # Ok, I'd choose, but for right now.
  1908. log.info("Safe {0} index {1}".format(safe_places, self.place_start))
  1909. put_people = safe_places[self.place_start]
  1910. log.info("Use: {0}".format(put_people))
  1911. self.place_start += 1
  1912. if self.place_start >= len(safe_places):
  1913. self.place_start = 0
  1914. self.queue_player.put("{0}\r\rQ".format(self.cargo_index[put_people] + 1))
  1915. self.colonists += self.fetch_amount
  1916. self.state = 6
  1917. def planet_chosen(self, choice: str):
  1918. if choice.strip() == '':
  1919. self.deactivate()
  1920. else:
  1921. self.planet_number = int(choice)
  1922. if self.planet_number in self.planets:
  1923. # Ok, this'll work
  1924. self.planet_sector = self.planets[self.planet_number]['sector']
  1925. self.planet_name = self.planets[self.planet_number]['name']
  1926. if self.current_sector == self.planet_sector:
  1927. # We're here. Land
  1928. self.state = 4
  1929. self.queue_player.put("L")
  1930. else:
  1931. # Get moving!
  1932. self.state = 3
  1933. self.queue_player.put("{0}\r".format(self.planet_sector))
  1934. else:
  1935. self.deactivate()
  1936. def game_line(self, line):
  1937. log.info("line {0} : {1}".format(self.state, line))
  1938. if self.state == 1:
  1939. if 'Personal Planet Scan' in line:
  1940. self.state = 2
  1941. elif self.state == 2:
  1942. # Ok, we're in the planet scan part of this
  1943. # 10202 #3 Home of Bugz Class M, Earth Type No Citadel]
  1944. if '#' in line:
  1945. # Ok, we have a planet detail line.
  1946. detail, _, _ = line.partition('Class')
  1947. detail = detail.strip() # Sector #X Name of planet
  1948. sector, number, name = re.split(r'\s+', detail, 2)
  1949. sector = int(sector)
  1950. number = int(number[1:])
  1951. self.last_seen = number
  1952. self.planets[number] = {"sector": sector, "name": name}
  1953. log.info("Planet # {0} in {1} called {2}".format( number, sector, name))
  1954. if '---' in line:
  1955. number = self.last_seen
  1956. # Ok, take the last_seen number, and use it for this line
  1957. # [ Sector Planet Name Ore Org Equ Ore Org Equ Fighters Citadel]
  1958. # [ Shields Population -=Productions=- -=-=-=-=-On Hands-=-=-=-=- Credits]
  1959. # [ 10202 #3 Home of Bugz Class M, Earth Type No Citadel]
  1960. # [ --- (1M) 144 49 26 145 75 12 10 0]
  1961. details = re.split(r"\s+", line.strip())
  1962. # OK: Likely, I'm not going to use these numbers AT ALL.
  1963. self.planets[number]['population'] = details[1].replace('(', '').replace(')', '')
  1964. # Ok, there's going to have to be some sort of modifier (K, M, etc.)
  1965. # to these numbers. Beware!
  1966. self.planets[number]['ore'] = details[5]
  1967. self.planets[number]['org'] = details[6]
  1968. self.planets[number]['equ'] = details[7]
  1969. elif self.state == 4:
  1970. # Combat Control Computer under construction, 4 day(s) till complete.
  1971. if 'under construction, ' in line and 'day(s) till complete' in line:
  1972. # Ok, already building.
  1973. self.queue_game.put(self.nl + Boxes.alert("NO, NOT YET!") + self.nl)
  1974. self.queue_player.put("Q") # quit Planet menu.
  1975. self.deactivate()
  1976. return
  1977. # [ Item Colonists Colonists Daily Planet Ship Planet ]
  1978. # [Fuel Ore 0 1 0 0 0 1,000,000]
  1979. # [Organics 0 N/A 0 0 0 10,000]
  1980. # [Equipment 0 500 0 125 125 100,000]
  1981. # [Fighters N/A N/A 0 0 400 1,000,000]
  1982. items = ['Fuel Ore', 'Organics', 'Equipment']
  1983. for i in items:
  1984. if line.startswith(i):
  1985. cargo = line[0].upper()
  1986. work = line.replace(',', '')
  1987. if i == 'Fuel Ore':
  1988. work = work.replace(i, 'Fuel')
  1989. self.colonists = 0
  1990. self.cargo = {}
  1991. self.need = {}
  1992. self.ship_cargo = {}
  1993. self.place_people = {}
  1994. parts = re.split(r'\s+', work)
  1995. log.info("parts: {0}".format(parts))
  1996. c = int(parts[1])
  1997. planet_has = int(parts[4])
  1998. self.colonists += c
  1999. self.cargo[cargo] = planet_has
  2000. self.ship_cargo[cargo] = int(parts[5])
  2001. # Boolean, can we place people here? If N/A, then don't!
  2002. self.place_people[cargo] = parts[2] != 'N/A'
  2003. self.place_start = 0
  2004. elif self.state == 5:
  2005. # [Planet command (?=help) [D] C]
  2006. # [Be patient, your Citadel is not yet finished.]
  2007. if line.startswith('Be patient, your Citadel is not yet finished.'):
  2008. # Ah HA!
  2009. self.queue_game.put(self.nl + Boxes.alert("NO, NOT YET!") + self.nl)
  2010. self.queue_player.put("Q") # quit Planet menu.
  2011. self.deactivate()
  2012. else:
  2013. items = ['Colonists', 'Fuel Ore', 'Organics', 'Equipment']
  2014. work = line.replace(',', '').replace(' units of', '').strip()
  2015. # 800,000 Colonists to support the construction,
  2016. # 500 units of Fuel Ore,
  2017. # 300 units of Organics,
  2018. # 600 units of Equipment and
  2019. for i in items:
  2020. if i in line:
  2021. count = int(work.split()[0])
  2022. k = i[0].upper()
  2023. # keep colonists in same units.
  2024. if k == 'C':
  2025. count //= 1000
  2026. self.need[k] = count
  2027. elif self.state == 8:
  2028. if re.match(r'How many groups of Colonists do you want to take \(\[\d+\] empty holds\) \?', line):
  2029. # Ok, how many holds?
  2030. _, _, holds = line.partition('[')
  2031. holds, _, _ = holds.partition(']')
  2032. self.fetch_amount = int(holds)
  2033. def __del__(self):
  2034. log.debug("PlanetUpScript {0} RIP".format(self))
  2035. def whenDone(self):
  2036. self.defer = defer.Deferred()
  2037. # Call this to chain something after we exit.
  2038. return self.defer
  2039. def deactivate(self):
  2040. self.state = 0
  2041. if not self.defer is None:
  2042. # We have something, so:
  2043. self.game.to_player = self.to_player
  2044. self.observer.load(self.save)
  2045. self.save = None
  2046. self.defer.callback(1)
  2047. self.defer = None
  2048. else:
  2049. # Still "exit" out.
  2050. self.game.to_player = self.to_player
  2051. self.observer.load(self.save)
  2052. def player(self, chunk):
  2053. """ Data from player (in bytes). """
  2054. chunk = chunk.decode("latin-1", "ignore")
  2055. key = chunk.upper()
  2056. log.warn("PlanetUpScript.player({0}) : I AM stopping...(user input)".format(key))
  2057. if not self.defer is None:
  2058. # We have something, so:
  2059. self.game.to_player = self.to_player
  2060. self.observer.load(self.save)
  2061. self.save = None
  2062. self.defer.errback(Exception("User Abort"))
  2063. self.defer = None
  2064. else:
  2065. # Still "exit" out.
  2066. self.game.to_player = self.to_player
  2067. self.observer.load(self.save)
  2068. class ProxyMenu(object):
  2069. """ Display ProxyMenu
  2070. Example:
  2071. from flexible import ProxyMenu
  2072. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", prompt):
  2073. menu = ProxyMenu(self.game)
  2074. """
  2075. def __init__(self, game):
  2076. self.nl = "\n\r"
  2077. self.c = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  2078. self.r = Style.RESET_ALL
  2079. self.c1 = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  2080. self.c2 = merge(Style.NORMAL + Fore.CYAN + Back.BLUE)
  2081. # self.portdata = None
  2082. self.game = game
  2083. self.queue_game = game.queue_game
  2084. self.observer = game.observer
  2085. # Am I using self or game? (I think I want game, not self.)
  2086. # if hasattr(self.game, "portdata"):
  2087. # self.portdata = self.game.portdata
  2088. # else:
  2089. # self.portdata = {}
  2090. # if hasattr(self.game, 'warpdata'):
  2091. # self.warpdata = self.game.warpdata
  2092. # else:
  2093. # self.warpdata = {}
  2094. if hasattr(self.game, 'trade_report'):
  2095. self.trade_report = self.game.trade_report
  2096. else:
  2097. self.trade_report = []
  2098. # Yes, at this point we would activate
  2099. self.prompt = game.buffer
  2100. self.save = self.observer.save()
  2101. self.observer.connect("player", self.player)
  2102. # If we want it, it's here.
  2103. self.defer = None
  2104. self.keepalive = task.LoopingCall(self.awake)
  2105. self.keepalive.start(30)
  2106. self.menu()
  2107. def __del__(self):
  2108. log.debug("ProxyMenu {0} RIP".format(self))
  2109. def whenDone(self):
  2110. self.defer = defer.Deferred()
  2111. # Call this to chain something after we exit.
  2112. return self.defer
  2113. def menu(self):
  2114. box = Boxes(30, color=self.c)
  2115. self.queue_game.put(box.top())
  2116. text = self.c + "{0:^30}".format("TradeWars Proxy Active")
  2117. text = text.replace('Active', BLINK + 'Active' + Style.RESET_ALL + self.c)
  2118. self.queue_game.put(box.row(text))
  2119. self.queue_game.put(box.middle())
  2120. def menu_item(ch: str, desc: str):
  2121. row = self.c1 + " {0} {1}- {2}{3:25}".format(ch, self.c2, self.c1, desc)
  2122. self.queue_game.put(box.row(row))
  2123. # self.queue_game.put(
  2124. # " " + self.c1 + ch + self.c2 + " - " + self.c1 + desc + self.nl
  2125. # )
  2126. menu_item("C", "Configuration ({0})".format(len(self.game.gamedata.config)))
  2127. menu_item("D", "Display Report again")
  2128. menu_item("E", "Export Data (Save)")
  2129. # menu_item("Q", "Quest")
  2130. menu_item("P", "Port CIM Report ({0})".format(len(self.game.gamedata.ports)))
  2131. menu_item("W", "Warp CIM Report ({0})".format(len(self.game.gamedata.warps)))
  2132. menu_item("T", "Trading Report")
  2133. menu_item("S", "Scripts")
  2134. menu_item("X", "eXit")
  2135. self.queue_game.put(box.bottom())
  2136. self.queue_game.put(" " + self.c + "-=>" + self.r + " ")
  2137. def awake(self):
  2138. log.info("ProxyMenu.awake()")
  2139. self.game.queue_player.put(" ")
  2140. def port_report(self, portdata: dict):
  2141. # self.portdata = portdata
  2142. # self.game.portdata = portdata
  2143. self.queue_game.put("Loaded {0} ports.".format(len(portdata)) + self.nl)
  2144. self.welcome_back()
  2145. def warp_report(self, warpdata: dict):
  2146. # self.warpdata = warpdata
  2147. # self.game.warpdata = warpdata
  2148. self.queue_game.put("Loaded {0} sectors.".format(len(warpdata)) + self.nl)
  2149. self.welcome_back()
  2150. def make_trade_report(self):
  2151. log.debug("make_trade_report()")
  2152. ok_trades = []
  2153. best_trades = []
  2154. show_best = self.game.gamedata.get_config('Display_Best', 'Y').upper()[0] == 'Y'
  2155. show_ok = self.game.gamedata.get_config('Display_Ok', 'N').upper()[0] == 'Y'
  2156. show_limit = self.game.gamedata.get_config('Display_Percent', '90')
  2157. update_config = False
  2158. try:
  2159. show_limit = int(show_limit)
  2160. except ValueError:
  2161. show_limit = 90
  2162. update_config = True
  2163. if show_limit < 0:
  2164. show_limit = 0
  2165. update_config = True
  2166. elif show_limit > 100:
  2167. show_limit = 100
  2168. update_config = True
  2169. if update_config:
  2170. self.game.gamedata.set_config('Display_Percent', show_limit)
  2171. # for sector, pd in self.game.gamedata.ports.items():
  2172. for sector in sorted(self.game.gamedata.ports.keys()):
  2173. pd = self.game.gamedata.ports[sector]
  2174. if not GameData.port_burnt(pd):
  2175. pc = pd['class']
  2176. # Ok, let's look into it.
  2177. if not sector in self.game.gamedata.warps:
  2178. continue
  2179. warps = self.game.gamedata.warps[sector]
  2180. for w in warps:
  2181. # Verify that we have that warp's info, and that the sector is in it.
  2182. # (We can get back from it)
  2183. if w in self.game.gamedata.warps and sector in self.game.gamedata.warps[w]:
  2184. # Ok, we can get there -- and get back!
  2185. if w > sector and w in self.game.gamedata.ports and not GameData.port_burnt(self.game.gamedata.ports[w]):
  2186. # it is > and has a port.
  2187. wd = self.game.gamedata.ports[w]
  2188. wc = wd['class']
  2189. # 1: "BBS",
  2190. # 2: "BSB",
  2191. # 3: "SBB",
  2192. # 4: "SSB",
  2193. # 5: "SBS",
  2194. # 6: "BSS",
  2195. # 7: "SSS",
  2196. # 8: "BBB",
  2197. if pc in (1,5) and wc in (2,4):
  2198. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  2199. if data:
  2200. best_trades.append(data)
  2201. # best_trades.append( "{0:5} -=- {1:5}".format(sector, w))
  2202. elif pc in (2,4) and wc in (1,5):
  2203. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  2204. if data:
  2205. best_trades.append(data)
  2206. # best_trades.append( "{0:5} -=- {1:5}".format(sector, w))
  2207. elif GameData.port_trading(pd['port'], self.game.gamedata.ports[w]['port']):
  2208. # ok_trades.append( "{0:5} -=- {1:5}".format(sector,w))
  2209. data = self.game.gamedata.port_trade_show(sector, w, show_limit)
  2210. if data:
  2211. ok_trades.append(data)
  2212. yield
  2213. if show_best:
  2214. self.trade_report.append("Best Trades: (org/equ)")
  2215. self.trade_report.extend(best_trades)
  2216. if show_ok:
  2217. self.trade_report.append("Ok Trades:")
  2218. self.trade_report.extend(ok_trades)
  2219. if not show_best and not show_ok:
  2220. self.queue_game.put(Boxes.alert("You probably want to choose something to display in configuration!", base="red"))
  2221. # self.queue_game.put("BEST TRADES:" + self.nl + self.nl.join(best_trades) + self.nl)
  2222. # self.queue_game.put("OK TRADES:" + self.nl + self.nl.join(ok_trades) + self.nl)
  2223. def get_display_maxlines(self):
  2224. show_maxlines = self.game.gamedata.get_config('Display_Maxlines', '0')
  2225. try:
  2226. show_maxlines = int(show_maxlines)
  2227. except ValueError:
  2228. show_maxlines = 0
  2229. if show_maxlines <= 0:
  2230. show_maxlines = None
  2231. return show_maxlines
  2232. def show_trade_report(self, *_):
  2233. show_maxlines = self.get_display_maxlines()
  2234. self.game.trade_report = self.trade_report
  2235. for t in self.trade_report[:show_maxlines]:
  2236. self.queue_game.put(t + self.nl)
  2237. self.queue_game.put(self.nl + Boxes.alert("Proxy done.", base="green"))
  2238. self.observer.load(self.save)
  2239. self.save = None
  2240. self.keepalive = None
  2241. self.prompt = None
  2242. # self.welcome_back()
  2243. def player(self, chunk: bytes):
  2244. """ Data from player (in bytes). """
  2245. chunk = chunk.decode("latin-1", "ignore")
  2246. key = chunk.upper()
  2247. log.debug("ProxyMenu.player({0})".format(key))
  2248. # Stop the keepalive if we are activating something else
  2249. # or leaving...
  2250. self.keepalive.stop()
  2251. if key == "T":
  2252. self.queue_game.put(self.c + key + self.r + self.nl)
  2253. # Trade Report
  2254. # do we have enough information to do this?
  2255. # if not hasattr(self.game, 'portdata') and not hasattr(self.game, 'warpdata'):
  2256. # self.queue_game.put("Missing portdata and warpdata." + self.nl)
  2257. # elif not hasattr(self.game, 'portdata'):
  2258. # self.queue_game.put("Missing portdata." + self.nl)
  2259. # elif not hasattr(self.game, 'warpdata'):
  2260. # self.queue_game.put("Missing warpdata." + self.nl)
  2261. # else:
  2262. if True:
  2263. # Yes, so let's start!
  2264. self.trade_report = []
  2265. d = coiterate(self.make_trade_report())
  2266. d.addCallback(self.show_trade_report)
  2267. return
  2268. elif key == "P":
  2269. self.queue_game.put(self.c + key + self.r + self.nl)
  2270. self.game.gamedata.reset_ports()
  2271. # Activate CIM Port Report
  2272. report = CIMPortReport(self.game)
  2273. d = report.whenDone()
  2274. d.addCallback(self.port_report)
  2275. d.addErrback(self.welcome_back)
  2276. return
  2277. elif key == "W":
  2278. self.queue_game.put(self.c + key + self.r + self.nl)
  2279. self.game.gamedata.reset_warps()
  2280. # Activate CIM Warp Report
  2281. report = CIMWarpReport(self.game)
  2282. d = report.whenDone()
  2283. d.addCallback(self.warp_report)
  2284. d.addErrback(self.welcome_back)
  2285. return
  2286. elif key == "S":
  2287. self.queue_game.put(self.c + key + self.r + self.nl)
  2288. # Scripts
  2289. self.activate_scripts_menu()
  2290. return
  2291. elif key == "D":
  2292. self.queue_game.put(self.c + key + self.r + self.nl)
  2293. # (Re) Display Trade Report
  2294. show_maxlines = self.get_display_maxlines()
  2295. if self.trade_report:
  2296. for t in self.trade_report[:show_maxlines]:
  2297. self.queue_game.put(t + self.nl)
  2298. self.queue_game.put(self.nl + Boxes.alert("Proxy done.", base="green"))
  2299. self.observer.load(self.save)
  2300. self.save = None
  2301. self.keepalive = None
  2302. self.prompt = None
  2303. return
  2304. else:
  2305. self.queue_game.put("Missing trade_report." + self.nl)
  2306. elif key == 'E':
  2307. self.queue_game.put(self.c + key + self.r + self.nl)
  2308. self.queue_game.put(Boxes.alert("Saving..."))
  2309. then_do = coiterate(self.game.gamedata.save())
  2310. then_do.addCallback(self.welcome_back)
  2311. return
  2312. elif key == "C":
  2313. self.queue_game.put(self.c + key + self.r + self.nl)
  2314. self.activate_config_menu()
  2315. return
  2316. # self.queue_game.put(pformat(self.portdata).replace("\n", "\n\r") + self.nl)
  2317. # self.queue_game.put(pformat(self.warpdata).replace("\n", "\n\r") + self.nl)
  2318. elif key == "Q":
  2319. self.queue_game.put(self.c + key + self.r + self.nl)
  2320. # This is an example of chaining PlayerInput prompt calls.
  2321. ask = PlayerInput(self.game)
  2322. d = ask.prompt("What is your quest?", 40, name="quest", abort_blank=True)
  2323. # Display the user's input
  2324. d.addCallback(ask.output)
  2325. d.addCallback(
  2326. lambda ignore: ask.prompt(
  2327. "What is your favorite color?", 10, name="color"
  2328. )
  2329. )
  2330. d.addCallback(ask.output)
  2331. d.addCallback(
  2332. lambda ignore: ask.prompt(
  2333. "What is the meaning of the squirrel?",
  2334. 12,
  2335. name="squirrel",
  2336. digits=True,
  2337. )
  2338. )
  2339. d.addCallback(ask.output)
  2340. def show_values(show):
  2341. log.debug(show)
  2342. self.queue_game.put(pformat(show).replace("\n", "\n\r") + self.nl)
  2343. d.addCallback(lambda ignore: show_values(ask.keep))
  2344. d.addCallback(self.welcome_back)
  2345. # On error, just return back
  2346. # This doesn't seem to be getting called.
  2347. # d.addErrback(lambda ignore: self.welcome_back)
  2348. d.addErrback(self.welcome_back)
  2349. return
  2350. elif key == "X":
  2351. self.queue_game.put(self.c + key + self.r + self.nl)
  2352. self.queue_game.put(Boxes.alert("Proxy done.", base="green"))
  2353. self.observer.load(self.save)
  2354. self.save = None
  2355. # It isn't running (NOW), so don't try to stop it.
  2356. # self.keepalive.stop()
  2357. self.keepalive = None
  2358. # Ok, this is a HORRIBLE idea, because the prompt might be
  2359. # outdated.
  2360. # self.queue_game.put(self.prompt)
  2361. self.prompt = None
  2362. # Send '\r' to re-display the prompt
  2363. # instead of displaying the original one.
  2364. self.game.queue_player.put("d")
  2365. # Were we asked to do something when we were done here?
  2366. if self.defer:
  2367. reactor.CallLater(0, self.defer.callback)
  2368. # self.defer.callback()
  2369. self.defer = None
  2370. return
  2371. self.keepalive.start(30, True)
  2372. self.menu()
  2373. def activate_config_menu(self):
  2374. self.observer.disconnect("player", self.player)
  2375. self.observer.connect("player", self.config_player)
  2376. self.config_menu()
  2377. def deactivate_config_menu(self, *data):
  2378. log.warn("deactivate_config_menu ({0})".format(data))
  2379. self.observer.disconnect("player", self.config_player)
  2380. self.observer.connect("player", self.player)
  2381. self.welcome_back()
  2382. def activate_scripts_menu(self):
  2383. self.observer.disconnect("player", self.player)
  2384. self.observer.connect("player", self.scripts_player)
  2385. self.scripts_menu()
  2386. def option_entry(self, entry):
  2387. if len(entry) > 0:
  2388. # Ok, they gave us something
  2389. self.game.gamedata.set_config(self.option_select, entry.strip())
  2390. else:
  2391. self.queue_game.put("Edit aborted." + self.nl)
  2392. self.config_menu()
  2393. def option_input(self, option):
  2394. if len(option) > 0:
  2395. option = int(option)
  2396. if option in self.config_opt:
  2397. # Ok, it's a valid option!
  2398. self.option_select = self.config_opt[option]
  2399. ask = PlayerInput(self.game)
  2400. d = ask.prompt("Change {0} to?".format(self.option_select), 18)
  2401. d.addCallback(self.option_entry)
  2402. # d.addErrback(self.config_menu)
  2403. else:
  2404. self.queue_game.put("Unknown option, sorry." + self.nl)
  2405. self.config_menu()
  2406. else:
  2407. # Aborted
  2408. self.config_menu()
  2409. def config_player(self, chunk: bytes):
  2410. """ Data from player (in bytes). """
  2411. chunk = chunk.decode("latin-1", "ignore")
  2412. key = chunk.upper()
  2413. if key == 'C':
  2414. self.queue_game.put(self.c + key + self.r + self.nl)
  2415. self.game.gamedata.config = {}
  2416. elif key == 'E':
  2417. self.queue_game.put(self.c + key + self.r + self.nl)
  2418. ask = PlayerInput(self.game)
  2419. d = ask.prompt("Which to edit?", 4, name='option', abort_blank=True, digits=True)
  2420. d.addCallback(self.option_input)
  2421. d.addErrback(self.config_menu)
  2422. return
  2423. elif key in ('1','2','3','4','5','6','7','8','9'):
  2424. self.queue_game.put(self.c + key + self.r + self.nl)
  2425. option = int(key)
  2426. if option in self.config_opt:
  2427. # Ok, it's a valid option!
  2428. self.option_select = self.config_opt[option]
  2429. ask = PlayerInput(self.game)
  2430. d = ask.prompt("Change {0} to?".format(self.option_select), 18)
  2431. d.addCallback(self.option_entry)
  2432. # d.addErrback(self.config_menu)
  2433. return
  2434. else:
  2435. self.queue_game.put("Unknown option, sorry." + self.nl)
  2436. elif key == 'X':
  2437. self.queue_game.put(self.c + key + self.r + self.nl)
  2438. self.deactivate_config_menu()
  2439. return
  2440. else:
  2441. self.queue_game.put(self.c + "?" + self.r + self.nl)
  2442. self.config_menu()
  2443. def config_menu(self, *_):
  2444. titlecolor = merge(Style.BRIGHT + Fore.CYAN + Back.BLUE)
  2445. tc = merge(Style.BRIGHT + Fore.YELLOW + Back.BLUE)
  2446. c1 = merge(Style.BRIGHT + Fore.WHITE + Back.BLUE)
  2447. c2 = merge(Style.BRIGHT + Fore.CYAN + Back.BLUE)
  2448. #box = Boxes(44, color=titlecolor)
  2449. box = Boxes(44, color=tc)
  2450. self.queue_game.put(box.top())
  2451. #self.queue_game.put(box.row(titlecolor + "{0:^44}".format("Configuration")))
  2452. self.queue_game.put(box.row(tc + "{0:^44}".format("Configuration")))
  2453. self.queue_game.put(box.middle())
  2454. def config_option(index, key, value):
  2455. row = "{0}{1:2} {2:19}{3}{4:<20}".format(c1, index, key, c2, value)
  2456. self.queue_game.put(box.row(row))
  2457. def menu_item(ch, desc):
  2458. row = "{0} {1} {2}-{3} {4:39}".format(c1, ch, c2, c1, desc)
  2459. # self.queue_game.put(
  2460. # " " + c1 + ch + c2 + " - " + c1 + desc + self.nl
  2461. # )
  2462. self.queue_game.put(box.row(row))
  2463. index = 1
  2464. self.config_opt = {}
  2465. for k in sorted(self.game.gamedata.config.keys()):
  2466. # for k, v in self.game.gamedata.config.items():
  2467. v = self.game.gamedata.config[k]
  2468. self.config_opt[index] = k
  2469. config_option(index, k, v)
  2470. index += 1
  2471. self.queue_game.put(box.middle())
  2472. menu_item("C", "Clear Config")
  2473. menu_item("E", "Edit Item")
  2474. menu_item("X", "eXit")
  2475. self.queue_game.put(box.bottom())
  2476. self.queue_game.put(" " + tc + "-=>" + self.r + " ")
  2477. def deactivate_scripts_menu(self, *data):
  2478. log.warn("deactivate_scripts_menu ({0})".format(data))
  2479. self.observer.disconnect("player", self.scripts_player)
  2480. self.observer.connect("player", self.player)
  2481. # Did they request exit?
  2482. if len(data) > 0 and type(data[0]) == dict:
  2483. info = data[0]
  2484. if 'exit' in info and info['exit']:
  2485. log.warn("exit proxy...")
  2486. # Exit Proxy Code
  2487. self.queue_game.put(self.nl + Boxes.alert("Proxy done.", base="green",style=3))
  2488. self.observer.load(self.save)
  2489. self.save = None
  2490. # It isn't running (NOW), so don't try to stop it.
  2491. # self.keepalive.stop()
  2492. self.keepalive = None
  2493. # Ok, this is a HORRIBLE idea, because the prompt might be
  2494. # outdated.
  2495. # self.queue_game.put(self.prompt)
  2496. self.prompt = None
  2497. # I'm not sure where we are, we might not be at a prompt.
  2498. # let's check!
  2499. if re.match(r"Command \[TL=.* \(\?=Help\)\? :", self.game.getPrompt()):
  2500. # Send '\r' to re-display the prompt
  2501. # instead of displaying the original one.
  2502. self.game.queue_player.put("d")
  2503. # Were we asked to do something when we were done here?
  2504. if self.defer:
  2505. reactor.CallLater(0, self.defer.callback)
  2506. # self.defer.callback()
  2507. self.defer = None
  2508. return
  2509. log.warn("calling welcome_back")
  2510. self.welcome_back()
  2511. def scripts_menu(self, *_):
  2512. c1 = merge(Style.BRIGHT + Fore.CYAN)
  2513. c2 = merge(Style.NORMAL + Fore.CYAN)
  2514. box = Boxes(40, color=c1)
  2515. self.queue_game.put(box.top())
  2516. self.queue_game.put(box.row(c1 + "{0:^40}".format("Scripts")))
  2517. self.queue_game.put(box.middle())
  2518. def menu_item(ch, desc):
  2519. row = " {0}{1} {2}-{3} {4:35}".format(c1, ch, c2, c1, desc)
  2520. # self.queue_game.put(
  2521. # " " + c1 + ch + c2 + " - " + c1 + desc + self.nl
  2522. # )
  2523. self.queue_game.put(box.row(row))
  2524. menu_item("1", "Ports (Trades between two sectors)")
  2525. menu_item("!", "Terrorize Ports/Trades")
  2526. menu_item("2", "Explore (Strange new sectors)")
  2527. menu_item("3", "Space... the broken script...")
  2528. menu_item("4", "Upgrade Planet")
  2529. menu_item("X", "eXit")
  2530. self.queue_game.put(box.bottom())
  2531. self.queue_game.put(" " + c1 + "-=>" + self.r + " ")
  2532. def terror(self, *_):
  2533. log.debug("terror {0}".format(_))
  2534. loops = _[0]
  2535. if loops.strip() == '':
  2536. self.deactivate_scripts_menu()
  2537. else:
  2538. # Ok, we have something here, I think...
  2539. terror = ScriptTerror(self.game, self, int(loops))
  2540. d = terror.whenDone()
  2541. d.addCallback(self.deactivate_scripts_menu)
  2542. d.addErrback(self.deactivate_scripts_menu)
  2543. def scripts_player(self, chunk: bytes):
  2544. """ Data from player (in bytes). """
  2545. chunk = chunk.decode("latin-1", "ignore")
  2546. key = chunk.upper()
  2547. if key == '1':
  2548. self.queue_game.put(self.c + key + self.r + self.nl)
  2549. # Activate this magical event here
  2550. ports = ScriptPort(self.game)
  2551. d = ports.whenDone()
  2552. # d.addCallback(self.scripts_menu)
  2553. # d.addErrback(self.scripts_menu)
  2554. d.addCallback(self.deactivate_scripts_menu)
  2555. d.addErrback(self.deactivate_scripts_menu)
  2556. return
  2557. elif key == '!':
  2558. self.queue_game.put(self.c + key + self.r + self.nl)
  2559. ask = PlayerInput(self.game)
  2560. # This is TERROR, so do something!
  2561. ask.color(merge(Style.BRIGHT + Fore.WHITE + Back.RED))
  2562. ask.colorp(merge(Style.BRIGHT + Fore.YELLOW + Back.RED))
  2563. d = ask.prompt("How many loops of terror?", 4, name="loops", digits=True, abort_blank=True)
  2564. d.addCallback(self.terror, ask)
  2565. d.addErrback(self.deactivate_scripts_menu)
  2566. return
  2567. elif key == '2':
  2568. self.queue_game.put(self.c + key + self.r + self.nl)
  2569. explore = ScriptExplore(self.game)
  2570. d = explore.whenDone()
  2571. d.addCallback(self.deactivate_scripts_menu)
  2572. d.addErrback(self.deactivate_scripts_menu)
  2573. return
  2574. elif key == '3':
  2575. self.queue_game.put(self.c + key + self.r + self.nl)
  2576. space = ScriptSpace(self.game)
  2577. d = space.whenDone()
  2578. d.addCallback(self.deactivate_scripts_menu)
  2579. d.addErrback(self.deactivate_scripts_menu)
  2580. return
  2581. elif key == '4':
  2582. self.queue_game.put(self.c + key + self.r + self.nl)
  2583. upgrade = PlanetUpScript(self.game)
  2584. d = upgrade.whenDone()
  2585. d.addCallback(self.deactivate_scripts_menu)
  2586. d.addErrback(self.deactivate_scripts_menu)
  2587. return
  2588. elif key == 'X':
  2589. self.queue_game.put(self.c + key + self.r + self.nl)
  2590. self.deactivate_scripts_menu()
  2591. return
  2592. else:
  2593. self.queue_game.put(self.c + "?" + self.r + self.nl)
  2594. self.scripts_menu()
  2595. def welcome_back(self, *_):
  2596. log.debug("welcome_back")
  2597. self.keepalive.start(30, True)
  2598. self.menu()