Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement expand extension interface and Cell struct #135

Merged
merged 2 commits into from
May 28, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
128 changes: 128 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -124,3 +124,131 @@ This is achieved by using Julia's native package extension mechanism. You can
find all the current package integrations in the `src/QuartoNotebookWorker/ext`
folder. Typically this is done via adding function hooks within the `__init__`
method of the extension that run at different points during notebook execution.

### Package Extensions

As discussed above `QuartoNotebookWorker` is implemented as a full Julia package
rather than just a `Module` loaded into the worker processes. This allows for
any package to extend the functionality provided by the worker via Julia's
[package extension mechanism][package-extensions]. For example, given a package
called `PackageName` you could create a new package extension in
`PackageName/ext/PackageNameQuartoNotebookWorkerExt.jl` with contents

[package-extensions]: https://pkgdocs.julialang.org/v1/creating-packages/#Conditional-loading-of-code-in-packages-(Extensions)

```julia
module PackageNameQuartoNotebookWorkerExt

import PackageName
import QuartoNotebookWorker

# ... Extension code here ...

end
```

and update the `Project.toml` file for the `PackageName` package to include the
following extension configuration:

```toml
[weakdeps]
QuartoNotebookWorker = "38328d9c-a911-4051-bc06-3f7f556ffeda"

[extensions]
PackageNameQuartoNotebookWorkerExt = "QuartoNotebookWorker"
```

With these additions whenever `PackageName` is loaded into a `.qmd` file that is
being run with `engine: julia` the extension code in the
`PackageNameQuartoNotebookWorkerExt` module will be loaded. Below are the
available interfaces that are can be extended.

#### `expand`

The `expand` function is used to inform `QuartoNotebookWorker` that a specific
Julia type should not be rendered and instead should be converted into a series
of notebook cells that are themselves evaluated and rendered. This allows for
notebooks to generate a dynamic number of cells based on runtime information
computed within the notebook rather than just the static cells of the original
notebook source. In some sense you can think of this as macros for notebooks:
code that generates code.
MichaelHatherly marked this conversation as resolved.
Show resolved Hide resolved

The below example shows how to create a `Replicate` type that will be expanded
into `n` cells of the same value.

```julia
module PackageNameQuartoNotebookWorkerExt

import PackageName
import QuartoNotebookWorker

function QuartoNotebookWorker.expand(r::PackageName.Replicate)
# Return a list of notebook `Cell`s to be rendered.
return [QuartoNotebookWorker.Cell(r.value) for _ in 1:r.n]
end

end
```

Where `PackageName` itself defines the `Replicate` type as

```julia
module PackageName

export Replicate

struct Replicate
value
n::Int
end

end
```

The `Cell` type takes a value, which can be any Julia type. If it is a
`Function` then the result of the `Cell` will be the result of calling the
`value()`, including any printing to `stdout` and `stderr` that may occur during
the call. If it is any other type then the result of the `Cell` will be the
value itself.

> [!NOTE]
>
> To return a `Function` itself as the output of the `Cell` you can wrap it
> with `Returns(func)`, which will then not call `func`.

Optional `code` keyword allows fake source code for the cell to be set, which
will be rendered by `quarto`. Note that the source code is never parsed or
evaluated. Additionally the `options` keyword allows for defining cell options
that will be passed to `quarto` to control cell rendering such as captions,
layout, etc.

Within a `.qmd` file you can then use the `Replicate` type as follows:

````qmd
```{julia}
using PackageName
```

Generate two cells that each output `"Hello"` as their returned value.

```{julia}
Replicate("Hello", 2)
```

Next we generate three cells that each push the current `DateTime` to a shared
`state` vector, print `"World"` to `stdout` and then return the entire `state`
for rendering. The `echo: false` option is used to suppress the output of the
original cell itself.

```{julia}
#| echo: false
import Dates
let state = []
Replicate(3) do
push!(state, Dates.now())
println("World")
return state
end
end
```
````
7 changes: 7 additions & 0 deletions src/QuartoNotebookWorker/src/QuartoNotebookWorker.jl
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
module QuartoNotebookWorker

# Exports:

export Cell
export expand


walk(x, _, outer) = outer(x)
walk(x::Expr, inner, outer) = outer(Expr(x.head, map(inner, x.args)...))
postwalk(f, x) = walk(x, x -> postwalk(f, x), f)
Expand Down Expand Up @@ -83,6 +89,7 @@ include("InlineDisplay.jl")
include("NotebookState.jl")
include("NotebookInclude.jl")
include("refresh.jl")
include("cell_expansion.jl")
include("render.jl")
include("utilities.jl")
include("ojs_define.jl")
Expand Down
86 changes: 86 additions & 0 deletions src/QuartoNotebookWorker/src/cell_expansion.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
"""
struct Cell
Cell(content; code = nothing, options = Dict())

`content` is either a callable object, which should have a 0-argument method
which, when called, represents the evaluation of the cell, or any other
non-callable object, which will simply be treated as the output value of the
cell. Should you wish to use a callable object as the output of a cell, rather
than calling it, you can wrap it in a `Returns` object.

`code` is the mock source code of the cell, and is not parsed or evaluated, but
will be rendered in the final output generated by `quarto`. When `code` is not
provided then the code block is hidden with `echo: false`. `options` represents
the cell options for the mock cell and will impact the rendering of the final
output.
"""
struct Cell
thunk::Base.Callable
code::String
options::Dict{String,Any}

function Cell(object; code::Union{String,Nothing} = nothing, options::Dict = Dict())
thunk = _make_thunk(object)
if isnothing(code)
options["echo"] = false
code = ""

Check warning on line 26 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L22-L26

Added lines #L22 - L26 were not covered by tests
end
new(thunk, code, options)

Check warning on line 28 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L28

Added line #L28 was not covered by tests
end
end

_make_thunk(c::Base.Callable) = c

Check warning on line 32 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L32

Added line #L32 was not covered by tests
# Returns is only available on 1.7 and upwards.
_make_thunk(other) = @static @isdefined(Returns) ? Returns(other) : () -> other

Check warning on line 34 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L34

Added line #L34 was not covered by tests

"""
expand(object::T) -> Vector{Cell}

Define the vector of `Cell`s that an object of type `T` should expand to. See
`Cell` documentation for more details on `Cell` creation.

This function is meant to be extended by 3rd party packages using Julia's
package extension mechanism. For example, within the `Project.toml` of a package
called `ExamplePackage` add the following:

```toml
[weakdeps]
QuartoNotebookWorker = "38328d9c-a911-4051-bc06-3f7f556ffeda"

[extensions]
ExamplePackageQuartoNotebookWorkerExt = "QuartoNotebookWorker"
```

Then create a file `ext/ExamplePackageQuartoNotebookWorkerExt.jl` with the the contents

```julia
module ExamplePackageQuartoNotebookWorkerExt

import ExamplePackage
import QuartoNotebookWorker

function QuartoNotebookWorker.expand(obj::ExamplePackage.ExampleType)
return [
QuartoNotebookWorker.Cell("This is the cell result."; code = "# Mock code goes here."),
QuartoNotebookWorker.Cell("This is a second cell."),
]
end

end
```
"""
expand(@nospecialize(_)) = nothing

Check warning on line 72 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L72

Added line #L72 was not covered by tests

struct CellExpansionError <: Exception
message::String
end

_is_expanded(@nospecialize(_), ::Nothing) = false
_is_expanded(@nospecialize(_), ::Vector{Cell}) = true
function _is_expanded(@nospecialize(original), @nospecialize(result))
throw(

Check warning on line 81 in src/QuartoNotebookWorker/src/cell_expansion.jl

View check run for this annotation

Codecov / codecov/patch

src/QuartoNotebookWorker/src/cell_expansion.jl#L78-L81

Added lines #L78 - L81 were not covered by tests
CellExpansionError(
"invalid cell expansion result for `expand(::$(typeof(original)))`. Expected the result to`Vector{Cell}`, got `$(typeof(result))`.",
MichaelHatherly marked this conversation as resolved.
Show resolved Hide resolved
),
)
end
Loading
Loading