Coming from Jupyter¶
If you're coming from Jupyter, here are a few tips to help you adapt to marimo notebooks.
Adapting to marimo's execution model¶
The biggest difference between marimo and Jupyter is the execution model.
A Jupyter notebook is a REPL: you execute blocks of code one at a time, and Jupyter has no understanding of how different blocks are related to each other. As a result a Jupyter notebook can easily accumulate "hidden state" (and hidden bugs) --- you might accidentally execute cells out of order, or you might run (or delete) a cell but forget to re-run cells that depended on its variables. Because of this, Jupyter notebooks suffer from a reproducibility crisis, with over a third of Jupyter notebooks on GitHub failing to reproduce.
Unlike Jupyter, marimo notebooks understand how different blocks of code are related to each other, modeling your code as a graph on cells based on variable declarations and references. This eliminates hidden state, and it's also what enables marimo notebooks to be reused as apps and scripts.
By default, if you run a cell in marimo, all other cells that read its variables run automatically. While this ensures that your code and outputs are in sync, it can take some time getting used to. Here are some tips and tools to help you adapt to marimo's execution model.
Configure marimo's runtime¶
Configure marimo's runtime to not autorun on startup or on cell execution.
Even when autorun is disabled, marimo still tracks dependencies across cells, marking dependents of a cell as stale when you run it. You can click a single button to run all your stale cells and bring your notebook back up-to-date.
Stop execution with mo.stop
¶
Use mo.stop
to stop a cell from executing if a condition
is met:
# if condition is True, the cell will stop executing after mo.stop() returns
mo.stop(condition)
# this won't be called if condition is True
expensive_function_call()
Use mo.stop()
in conjunction with
mo.ui.run_button()
to require a button press for
expensive cells:
Working with expensive notebooks¶
For more tips on adapting to marimo's execution model, see our guide on working with expensive notebooks.
Adapting to marimo's restriction on redefining variables¶
marimo "compiles" your notebook cells into a directed graph on cells, linked by variable declarations and references, reusing this graph to run your notebook as a script or app. For marimo's compilation to work, the same variable cannot be defined in multiple cells; otherwise, marimo wouldn't know what order to run cells in.
To adapt to the restriction, we suggest:
- Encapsulating code into functions when possible, to minimize the number of global variables.
- Prefixing temporary variables with an underscore (
_my_temporary
), which makes the variable local to a cell. - Mutating variables in the cell that defines them.
When working with dataframes, you might be used to redefining the same df
variable in multiple cells. That won't work in marimo. Instead, try merging
the cells into a single cell:
Don't do this:
Instead, do this:
If you do need to transform a dataframe across multiple cells, you can alias the dataframe:
Adapting to marimo's file format¶
marimo stores notebooks as Python, not JSON. This lets you version notebooks with git, execute them as scripts, and import named cells into other Python files. However, it does mean that your notebook outputs (e.g., plots) are not stored in the file.
If you'd like to keep a visual record of your notebook work, enable
the "Auto-download as HTML" setting, which will
periodically snapshot your notebook as HTML to a __marimo__
folder in the
notebook directory.
Converting Jupyter notebooks to marimo notebooks¶
Convert Jupyter notebooks to marimo notebooks at the command-line:
Adapting to the absence of magic commands¶
Because marimo notebooks are just Python (improving maintainability), marimo
doesn't support IPython magic commands or !
-prefixed console commands. Here
are some alternatives.
Run console commands with subprocess.run¶
To run a console command, use Python's subprocess.run:
Common magic commands replacements¶
Magic Command | Replacement |
---|---|
%cd | os.chdir() , see also mo.notebook_dir() |
%clear | Right-click or toggle the cell actions |
%debug | Python's built-in debugger: breakpoint() |
%env | os.environ |
%load | N/A - use Python imports |
%load_ext | N/A |
%autoreload | marimo's module autoreloader |
%matplotlib | marimo auto-displays plots |
%pwd | os.getcwd() |
%pip | Use marimo's built-in package management |
%who_ls | dir() , globals() , mo.refs() , mo.defs() |
%system | subprocess.run() |
%%time | time.perf_counter() or Python's timeit module |
%%timeit | Python's timeit module |
%%writefile | with open("file.txt", "w") as f: f.write() |
%%capture | mo.capture_stdout() , mo.capture_stderr() |
%%html | mo.Html() or mo.md() |
%%latex | mo.md(r'$$...$$') |
Installing packages with marimo's package manager¶
Use marimo's package management sidebar panel to install packages to your current environment. Learn more in our package management guide.
Interactive guide¶
This guide contains additional tips to help you adapt to marimo. Fun fact: the guide is itself a marimo notebook!