flexible.py 147 KB

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