A simple web application built in Rust using Axum for routing and Minijinja for templating. This project pulls content from Storyblok using a custom macro and renders dynamic pages with Jinja templates.
- Rust Web Server: Uses
axum
for building an asynchronous HTTP server. - Templating: Renders HTML pages using
minijinja
with custom layouts and page templates. - Dynamic Data: Fetches data from Storyblok through a custom
get_data!
macro defined insrc/macros.rs
. - Static Assets: Serves static files such as the CSS and favicon from
src/static/
.
- src/main.rs: Application entry point that starts the Axum server. See src/main.rs.
- src/environment.rs: Sets up the Jinja templating environment and loads templates. See src/environment.rs.
- src/pages/: Contains page modules for different routes:
- Home: src/pages/home/mod.rs and its template src/pages/home/index.jinja.
- Blog: src/pages/blog/mod.rs with the article submodule src/pages/blog/article/mod.rs and template src/pages/blog/article/index.jinja.
- Fallback: src/pages/fallback/mod.rs and its template src/pages/fallback/index.jinja.
- src/layout/: Contains the main layout template: src/layout/default.jinja.
- src/router/: Implements routing logic:
- Page Routes: src/router/page_routes.rs
- Static Source Routes: src/router/source_routes.rs
- static: Static files such as src/static/index.css and src/static/favicon.ico.
-
Rust: Ensure you have Rust installed (this project uses Rust edition 2021).
-
Environment Variables: Create a
.env
file in the project root with the following variables:ST_TOKEN=your_storyblok_token ST_BASE_URL=https://api.storyblok.com/v2/cdn/stories
-
Clone the repository:
git clone https://github.com/falcosan/ap cd ap
-
Install Dependencies: The project relies on crates defined in Cargo.toml. Cargo will automatically download these dependencies during the build.
For development, you can use the provided shell script:
./dev.sh
The server binds to 127.0.0.1:8000
. Open your browser at http://127.0.0.1:8000 to view the application.
To build the project in release mode, run:
cargo build --release
This project is licensed under the MIT License.
The extract_components!
custom macro defined in src/macros.rs
, recursively extracts JSON components matching a given target name from the provided JSON data.
This macro searches through a JSON structure (which is expected to be a serde_json::Value
) for objects
that have a "component"
key. When it finds such an object, it checks whether the associated value matches
the specified target name.
If the target name is "TextContent"
, the macro converts any Markdown text found under the "text"
key into HTML
using the pulldown_cmark
parser. The resulting HTML is then inserted back into the JSON object in place of the original text.
All matching components (with or without HTML conversion) are collected into a Vec<serde_json::Value>
and returned.
data
: An expression yielding aserde_json::Value
containing the JSON tree to search.name
: An expression yielding a string slice representing the target component's name.
// Suppose you have a JSON string representing your data:
let json_str = "{
"component": "TextContent",
"text": "# Hello World!"
}";
// Parse the string into a serde_json::Value
let data: serde_json::Value = serde_json::from_str(json_str).unwrap();
// Call the macro to extract components of type "TextContent"
let components = extract_components!(data, "TextContent");
// components now contains the content with the Markdown in "text" transformed into HTML.
// Further processing can be performed on this Vec<serde_json::Value> as needed.
- Make sure your project includes the
serde_json
andpulldown_cmark
crates. - Import the module that defines this macro before using it.
- The macro internally traverses both JSON arrays and objects. Non-object or non-array values are skipped.
- When working with
"TextContent"
components, the macro expects a"text"
key to be present in the object; otherwise, no HTML conversion is performed.
To use extract_components!
:
- Add the module (e.g.,
macros.rs
) containing the macro to your project. - Ensure that the module is imported where needed (using
mod macros;
or similar). - Invoke the macro by passing a
serde_json::Value
and a target component name as demonstrated above.