Skip to content

Commit

Permalink
implemented quantopedia command
Browse files Browse the repository at this point in the history
Fixes #177
  • Loading branch information
solodov committed Jun 27, 2024
1 parent 259bf19 commit ade7987
Show file tree
Hide file tree
Showing 3 changed files with 63 additions and 18 deletions.
56 changes: 47 additions & 9 deletions examples/quantum_rpg/main_loop.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,49 +12,71 @@
# See the License for the specific language governing permissions and
# limitations under the License.

from typing import List, Optional, Sequence
import enum
import io
import sys
import textwrap
from typing import List, Optional, Sequence

from . import ascii_art
from . import battle
from . import classes
from . import exceptions
from . import game_state
from . import input_helpers
from . import npcs
from . import world
from . import xp_utils
from .final_state_preparation import final_state_world


class Command(enum.Enum):
"""Command parsing utility.
class Error(Exception):
"""Base class for locally defined exceptions."""
pass

Currently only supports 'quit' but future expansion planned.
"""

class AmbiguousCommandError(Error):
"""Raised when entered command is ambiguous."""
pass


class Command(enum.Enum):
"""Enumeration of available commands."""

LOAD = "load"
LOOK = "look"
STATUS = "status"
SAVE = "save"
HELP = "help"
QUANTOPEDIA = "quantopedia"
QUIT = "quit"

@classmethod
def parse(cls, s: str) -> Optional["Command"]:
"""Parses a string as a Direction.
"""Parses a string as a command.
Allows prefixes, like 'e' to be parsed as EAST.
"""
if not s:
return None
lower_s = s.lower()
for cmd in Command:
candidates = []
for cmd in cls:
if cmd.value.startswith(lower_s):
return cmd
candidates.append(cmd)
if len(candidates) == 1:
return candidates[0]
if len(candidates) > 1:
raise AmbiguousCommandError
return None

@classmethod
def help(cls) -> str:
cmds = ["Available commands:"]
for cmd in Command:
cmds.append(f" {cmd.value}")
return "\n".join(cmds)


class MainLoop:
def __init__(self, world: world.World, state: game_state.GameState):
Expand Down Expand Up @@ -135,7 +157,14 @@ def loop(self, user_input: Optional[Sequence[str]] = None) -> None:
print(msg, file=self.file)
print_room_description = False
continue
input_cmd = Command.parse(current_input)
try:
input_cmd = Command.parse(current_input)
except AmbiguousCommandError:
print(f"Ambiguous command '{current_input}'.",
Command.help(),
file=self.file)
print_room_description = False
continue
if input_cmd == Command.QUIT:
return
elif input_cmd == Command.STATUS:
Expand All @@ -144,6 +173,15 @@ def loop(self, user_input: Optional[Sequence[str]] = None) -> None:
elif input_cmd == Command.HELP:
print(ascii_art.HELP, file=self.file)
print_room_description = False
elif input_cmd == Command.QUANTOPEDIA:
print(file=self.file)
for npc_class in npcs.Npc.__subclasses__():
print(npc_class.__name__, file=self.file)
print(
textwrap.indent(npc_class.quantopedia_entry(), " "),
file=self.file)
print(file=self.file)
print_room_description = False
elif input_cmd == Command.LOAD:
print(
"Paste the save file here to load the game from that point.",
Expand Down
4 changes: 2 additions & 2 deletions examples/quantum_rpg/main_loop_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -105,8 +105,8 @@ def example_world():

def test_parse_commands() -> None:
assert main_loop.Command.parse("x") is None
assert main_loop.Command.parse("q") is main_loop.Command.QUIT
assert main_loop.Command.parse("Q") is main_loop.Command.QUIT
assert main_loop.Command.parse("qui") is main_loop.Command.QUIT
assert main_loop.Command.parse("Qui") is main_loop.Command.QUIT
assert main_loop.Command.parse("Quit") is main_loop.Command.QUIT
assert main_loop.Command.parse("quit") is main_loop.Command.QUIT

Expand Down
21 changes: 14 additions & 7 deletions examples/quantum_rpg/npcs.py
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,8 @@ def quantopedia_index(self) -> int:
"""
return 0

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
"""Explanatory text to the players about the NPC."""
return "Nothing known about this NPC."

Expand All @@ -112,7 +113,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _FOAM_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Observers are known to frequent quantum events.\n"
"They will measure qubits in order to find out their values."
Expand All @@ -137,7 +139,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _FOAM_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Blue foam are the simplest kind of quantum errors. Blue foam\n"
"are usually found in the |0> state and can be measured.\n"
Expand Down Expand Up @@ -165,7 +168,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _FOAM_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Green foam are a simple kind of quantum error. Green foam\n"
"are usually found in the |0> state and can be measured immediately.\n"
Expand Down Expand Up @@ -195,7 +199,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _FOAM_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Red foam are a slightly more dangerous type of quantum error.\n"
"They are usually found in the |1> state and must be flipped\n"
Expand Down Expand Up @@ -224,7 +229,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _FOAM_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Purple foam are a combination of red and blue form.\n"
"They are found in a |+> state which is a combination of\n"
Expand Down Expand Up @@ -268,7 +274,8 @@ def act_on_enemy_qubit(self, enemy_qubit, action_choice, **kwargs) -> str:
def quantopedia_index(self) -> int:
return _HILLS_QUANTOPEDIA

def quantopedia_entry(self) -> str:
@classmethod
def quantopedia_entry(cls) -> str:
return (
"Schrödinger's cat are found in a superposition of zero and one.\n"
"This cat contains multiple qubits that are entangled so that all\n"
Expand Down

0 comments on commit ade7987

Please sign in to comment.