Multi-tape Turing Machine simulator in Python
Project description
mtm — Multi-tape Turing Machine simulator in Python
mtm is a small, reusable library for simulating multi-tape Turing Machines in Python.
It provides:
- A core
MultiTapeTuringMachinesimulator. - A
Tapeabstraction with infinite tapes in both directions. - A
Directionenum for head movements. - A
TuringMachineDefinitionclass to encapsulate reusable machine definitions. - A
consolehelper (animate_run) to visualize head movements as a terminal animation. - Ready-to-use example machines under
mtm.examples.
The goal is to make it easy to experiment with Turing Machines, build examples for complexity theory courses, and integrate the simulator into other projects (GUIs, notebooks, etc.).
Installation
From PyPI:
pip install multitape-tm
From the project root (where pyproject.toml lives):
python -m venv .venv
source .venv/bin/activate # on Windows: .venv\Scripts\activate
python -m pip install -e .
This installs mtm in editable mode, so changes to the source code are immediately reflected when you import the package.
Quick start
Basic 1-tape example
The smallest possible example: a 1-tape TM that moves right over a run of a’s and accepts when it hits blank.
from mtm import MultiTapeTuringMachine, Direction
# States and alphabets
states = {"q0", "q_accept"}
input_alphabet = {"a"}
tape_alphabet = {"a", "_"}
blank = "_"
start_state = "q0"
accept_states = {"q_accept"}
# Transition function:
# (state, (symbols_on_tapes, ...)) -> (new_state, (symbols_to_write, ...), (moves, ...))
transitions = {
# While reading 'a', move right
("q0", ("a",)): ("q0", ("a",), (Direction.RIGHT,)),
# When reaching blank, accept
("q0", ("_",)): ("q_accept", ("_",), (Direction.STAY,)),
}
tm = MultiTapeTuringMachine(
states=states,
input_alphabet=input_alphabet,
tape_alphabet=tape_alphabet,
blank=blank,
transitions=transitions,
start_state=start_state,
accept_states=accept_states,
reject_states=set(),
num_tapes=1,
initial_inputs=["aaa"],
)
result = tm.run()
print("Final status:", result)
print("Final configuration:", tm.get_configuration())
Animating the heads in the terminal
mtm.console.animate_run clears the screen every step and shows a live “animation” of the tape(s), with the head position indicated by [...].
from mtm import MultiTapeTuringMachine, Direction, animate_run
states = {"q0", "q_accept"}
input_alphabet = {"a"}
tape_alphabet = {"a", "_"}
blank = "_"
start_state = "q0"
accept_states = {"q_accept"}
transitions = {
("q0", ("a",)): ("q0", ("a",), (Direction.RIGHT,)),
("q0", ("_",)): ("q_accept", ("_",), (Direction.STAY,)),
}
tm = MultiTapeTuringMachine(
states=states,
input_alphabet=input_alphabet,
tape_alphabet=tape_alphabet,
blank=blank,
transitions=transitions,
start_state=start_state,
accept_states=accept_states,
reject_states=set(),
num_tapes=1,
initial_inputs=["aaa"],
)
animate_run(tm, max_steps=50, delay=0.2)
Output (sketch):
Multi-tape Turing Machine
========================================
Step : 3
State: q0
Tape 0: a a [a] _
(press Ctrl+C to stop)
Library overview
Core types
-
Tape-
Represents an infinite tape in both directions.
-
Stores only non-blank cells in a dictionary
index -> symbol. -
Key methods:
read() -> strwrite(symbol: str) -> Nonemove(direction: int) -> None__str__()gives a human-readable view with the head wrapped in[...].
-
-
Direction(enum)Direction.LEFT = -1Direction.RIGHT = 1Direction.STAY = 0
-
MultiTapeTuringMachine-
Simulates a TM with k tapes.
-
Constructor parameters:
-
states: iterable of states (typically strings). -
input_alphabet: symbols that may appear in the input. -
tape_alphabet: symbols that may appear on the tapes (must includeinput_alphabetandblank). -
blank: blank symbol. -
transitions: dict with keys and values:-
Key:
(state, (s1, ..., sk)) -
Value:
(new_state, (w1, ..., wk), (m1, ..., mk))wiare symbols to write.miare movements (Direction.LEFT,Direction.RIGHT,Direction.STAY, or-1, 0, 1).
-
-
start_state -
accept_states -
reject_states -
num_tapes -
initial_inputs: list of strings, one per tape.
-
-
Main methods:
-
step() -> str- Executes one step.
- Returns
"RUNNING","ACCEPT","REJECT"or"HALT".
-
run(max_steps: int = 10_000) -> str- Runs until halting, accepting/rejecting, or reaching
max_steps.
- Runs until halting, accepting/rejecting, or reaching
-
reset(initial_inputs: Optional[Sequence[str]] = None) -> None -
get_configuration() -> dict- Returns
{ "state": ..., "tapes": [...], "step": ... }.
- Returns
-
-
-
TuringMachineDefinition-
A dataclass that stores a reusable definition of a machine:
statesinput_alphabettape_alphabetblanktransitionsstart_stateaccept_statesreject_statesnum_tapes
-
Method:
create_machine(initial_inputs: Optional[Sequence[str]] = None) -> MultiTapeTuringMachine
This is intended for building reusable models: you define the structure once, and then create many instances with different inputs.
-
Using the built-in examples
The package includes several predefined Turing Machine definitions under mtm.examples.
All examples accept an input string and return a
MultiTapeTuringMachineinstance ready to run or animate.
1. Binary palindrome (2 tapes)
mtm.examples.palindrome_2tape implements a 2-tape TM that checks if a binary string is a palindrome.
from mtm.examples import build_palindrome_2tape_machine
from mtm import animate_run
tm = build_palindrome_2tape_machine("0110")
status = animate_run(tm, max_steps=200, delay=0.3)
print("Final status:", status)
- Tape 1: input.
- Tape 2: used for marks.
- Accepts if the string is a palindrome, rejects by lack of transitions.
2. Language aⁿ bⁿ cⁿ (1 tape)
mtm.examples.abc_equal implements a 1-tape TM for the language:
L = { aⁿ bⁿ cⁿ | n ≥ 1 }
from mtm.examples import build_abc_equal_machine
from mtm import animate_run
tm = build_abc_equal_machine("aabbcc") # n = 2
status = animate_run(tm, max_steps=200, delay=0.3)
print("Final status:", status)
The machine:
- Searches for the next unmarked
a, marks it asX. - Searches for the next unmarked
b, marks it asY. - Searches for the next unmarked
c, marks it asZ. - Returns to the left end and repeats.
- Accepts if no
aremain and there are no extraborc.
3. Language w%w (1 tape)
mtm.examples.ww_delimiter implements a 1-tape TM for:
L = { w % w | w ∈ {a, b}* }
from mtm.examples import build_ww_delimiter_machine
from mtm import animate_run
tm = build_ww_delimiter_machine("abba%abba")
status = animate_run(tm, max_steps=400, delay=0.3)
print("Final status:", status)
The machine:
- Scans the first
w, marking letters asXorY. - For each marked letter before
%, it looks for a matching unmarked letter after%. - When there are no unmarked letters before
%, it checks that no unmarked letters remain after%. - Accepts if and only if the two halves are identical.
Defining your own machines
You have two main options:
Option A: Use MultiTapeTuringMachine directly
This is the most direct, flexible way.
from mtm import MultiTapeTuringMachine, Direction
states = {"q0", "q_accept"}
input_alphabet = {"0", "1"}
tape_alphabet = {"0", "1", "_"}
blank = "_"
transitions = {
# Toy example: move right while reading 0/1, accept on blank
("q0", ("0",)): ("q0", ("0",), (Direction.RIGHT,)),
("q0", ("1",)): ("q0", ("1",), (Direction.RIGHT,)),
("q0", ("_",)): ("q_accept", ("_",), (Direction.STAY,)),
}
tm = MultiTapeTuringMachine(
states=states,
input_alphabet=input_alphabet,
tape_alphabet=tape_alphabet,
blank=blank,
transitions=transitions,
start_state="q0",
accept_states={"q_accept"},
reject_states=set(),
num_tapes=1,
initial_inputs=["0101"],
)
Option B: Subclass TuringMachineDefinition
Use this if you want a reusable library model that you can instantiate many times with different inputs (similar to the examples).
from __future__ import annotations
from typing import Dict, Set
from mtm import TuringMachineDefinition, MultiTapeTuringMachine, Direction
from mtm.machine import State, Symbol, TransitionKey, TransitionValue
class MyToyDefinition(TuringMachineDefinition):
"""
Example of a custom 1-tape TM definition.
Accepts any string of 0/1 that ends in 1.
"""
def __init__(self) -> None:
states: Set[State] = {"q0", "q_accept", "q_reject"}
input_alphabet: Set[Symbol] = {"0", "1"}
tape_alphabet: Set[Symbol] = {"0", "1", "_"}
blank: Symbol = "_"
transitions: Dict[TransitionKey, TransitionValue] = {
("q0", ("0",)): ("q0", ("0",), (Direction.RIGHT,)),
("q0", ("1",)): ("q0", ("1",), (Direction.RIGHT,)),
# On blank, last symbol was 1 if the head is to the right of a 1
# (here, for simplicity, we just accept on blank)
("q0", ("_",)): ("q_accept", ("_",), (Direction.STAY,)),
}
super().__init__(
states=states,
input_alphabet=input_alphabet,
tape_alphabet=tape_alphabet,
blank=blank,
transitions=transitions,
start_state="q0",
accept_states={"q_accept"},
reject_states={"q_reject"},
num_tapes=1,
)
# Helper function, similar to the ones in mtm.examples
def build_my_toy_machine(input_string: str) -> MultiTapeTuringMachine:
definition = MyToyDefinition()
return definition.create_machine(initial_inputs=[input_string])
Usage:
from my_module import build_my_toy_machine
from mtm import animate_run
tm = build_my_toy_machine("0101")
animate_run(tm)
This pattern is recommended if you are building a collection of Turing Machines for a course, a paper, or a larger project.
Project structure (for contributors)
src/
mtm/
__init__.py
tape.py # Tape + Direction
machine.py # MultiTapeTuringMachine
console.py # animate_run for terminal animation
definition.py # TuringMachineDefinition
examples/
__init__.py
palindrome_2tape.py # binary palindrome with 2 tapes
abc_equal.py # a^n b^n c^n
ww_delimiter.py # w%w
Local top-level demo scripts used for manual testing (not published) are kept outside src/ and ignored by git (see .gitignore).
License
This project is licensed under the Apache License 2.0.
You are free to use, modify, and distribute this code (including in commercial and closed-source projects) as long as you keep the copyright and license notices, and respect the terms of the Apache 2.0 license.
Roadmap / Ideas
- More built-in examples (unary addition, multiplication, etc.).
- Ready-made GUIs for step-by-step visualization.
- Export/import Turing Machines from JSON or YAML.
- Jupyter notebook helpers.
Contributions and suggestions are very welcome 🚀
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file multitape_tm-0.1.0.tar.gz.
File metadata
- Download URL: multitape_tm-0.1.0.tar.gz
- Upload date:
- Size: 18.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
bd3f4328df556a77d1814d222844f8d1a07a990832c151d77fbd6b3a23163e0f
|
|
| MD5 |
e4077ae92c0078ac1302ce9143bd9de7
|
|
| BLAKE2b-256 |
687be4ee313f6028468178f85516d24be65f2f2efb8582d1fdd3bd3bbd66bfbb
|
File details
Details for the file multitape_tm-0.1.0-py3-none-any.whl.
File metadata
- Download URL: multitape_tm-0.1.0-py3-none-any.whl
- Upload date:
- Size: 17.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5d46c5a8878cbc357f7b32efbbce694141b8b902c86f9b4af9f67319add4d70d
|
|
| MD5 |
9102513b1969dfda19ed0e5228b67b2b
|
|
| BLAKE2b-256 |
83424e5878aa933210f799ddf86be13d9db7f3f3876a696f99ac50b35085c108
|