Skip to content

Creating projects

fyn supports creating a project with fyn init.

When creating projects, fyn supports two basic templates: applications and libraries. By default, fyn will create a project for an application. The --lib flag can be used to create a project for a library instead.

Target directory

fyn will create a project in the working directory, or, in a target directory by providing a name, e.g., fyn init foo. The working directory can be modified with the --directory option, which will cause the target directory path to be interpreted relative to the specified working directory. If there's already a project in the target directory, i.e., if there's a pyproject.toml, fyn will exit with an error.

Applications

Application projects are suitable for web servers, scripts, and command-line interfaces.

Applications are the default target for fyn init, but can also be specified with the --app flag.

$ fyn init example-app

The project includes a pyproject.toml, a sample file (main.py), a readme, and a Python version pin file (.python-version).

$ tree example-app
example-app
├── .python-version
├── README.md
├── main.py
└── pyproject.toml

Note

Prior to v0.6.0, fyn created a file named hello.py instead of main.py.

The pyproject.toml includes basic metadata. It does not include a build system, it is not a package and will not be installed into the environment:

pyproject.toml
[project]
name = "example-app"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.11"
dependencies = []

The sample file defines a main function with some standard boilerplate:

main.py
def main():
    print("Hello from example-app!")


if __name__ == "__main__":
    main()

Python files can be executed with fyn run:

$ cd example-app
$ fyn run main.py
Hello from example-project!

Packaged applications

Many use-cases require a package. For example, if you are creating a command-line interface that will be published to PyPI or if you want to define tests in a dedicated directory.

The --package flag can be used to create a packaged application:

$ fyn init --package example-pkg

The source code is moved into a src directory with a module directory and an __init__.py file:

$ tree example-pkg
example-pkg
├── .python-version
├── README.md
├── pyproject.toml
└── src
    └── example_pkg
        └── __init__.py

A build system is defined, so the project will be installed into the environment:

pyproject.toml
[project]
name = "example-pkg"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.11"
dependencies = []

[project.scripts]
example-pkg = "example_pkg:main"

[build-system]
requires = ["fyn_build>=0.10.14,<0.11.0"]
build-backend = "fyn_build"

Tip

The --build-backend option can be used to request an alternative build system.

A command definition is included:

pyproject.toml
[project]
name = "example-pkg"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.11"
dependencies = []

[project.scripts]
example-pkg = "example_pkg:main"

[build-system]
requires = ["fyn_build>=0.10.14,<0.11.0"]
build-backend = "fyn_build"

The command can be executed with fyn run:

$ cd example-pkg
$ fyn run example-pkg
Hello from example-pkg!

Libraries

A library provides functions and objects for other projects to consume. Libraries are intended to be built and distributed, e.g., by uploading them to PyPI.

Libraries can be created by using the --lib flag:

$ fyn init --lib example-lib

Note

Using --lib implies --package. Libraries always require a packaged project.

As with a packaged application, a src layout is used. A py.typed marker is included to indicate to consumers that types can be read from the library:

$ tree example-lib
example-lib
├── .python-version
├── README.md
├── pyproject.toml
└── src
    └── example_lib
        ├── py.typed
        └── __init__.py

Note

A src layout is particularly valuable when developing libraries. It ensures that the library is isolated from any python invocations in the project root and that distributed library code is well separated from the rest of the project source.

A build system is defined, so the project will be installed into the environment:

pyproject.toml
[project]
name = "example-lib"
version = "0.1.0"
description = "Add your description here"
readme = "README.md"
requires-python = ">=3.11"
dependencies = []

[build-system]
requires = ["fyn_build>=0.10.14,<0.11.0"]
build-backend = "fyn_build"

Tip

You can select a different build backend template by using --build-backend with hatchling, fyn_build, flit-core, pdm-backend, setuptools, maturin, or scikit-build-core. An alternative backend is required if you want to create a library with extension modules.

The created module defines a simple API function:

__init__.py
def hello() -> str:
    return "Hello from example-lib!"

And you can import and execute it using fyn run:

$ cd example-lib
$ fyn run python -c "import example_lib; print(example_lib.hello())"
Hello from example-lib!

Projects with extension modules

Most Python projects are "pure Python", meaning they do not define modules in other languages like C, C++, FORTRAN, or Rust. However, projects with extension modules are often used for performance sensitive code.

Creating a project with an extension module requires choosing an alternative build system. fyn supports creating projects with the following build systems that support building extension modules:

Specify the build system with the --build-backend flag:

$ fyn init --build-backend maturin example-ext

Note

Using --build-backend implies --package.

The project contains a Cargo.toml and a lib.rs file in addition to the typical Python project files:

$ tree example-ext
example-ext
├── .python-version
├── Cargo.toml
├── README.md
├── pyproject.toml
└── src
    ├── lib.rs
    └── example_ext
        ├── __init__.py
        └── _core.pyi

Note

If using scikit-build-core, you'll see CMake configuration and a main.cpp file instead.

The Rust library defines a simple function:

src/lib.rs
use pyo3::prelude::*;

#[pymodule]
mod _core {
    use pyo3::prelude::*;

    #[pyfunction]
    fn hello_from_bin() -> String {
        "Hello from example-ext!".to_string()
    }
}

And the Python module imports it:

src/example_ext/__init__.py
from example_ext._core import hello_from_bin


def main() -> None:
    print(hello_from_bin())

The command can be executed with fyn run:

$ cd example-ext
$ fyn run example-ext
Hello from example-ext!

Important

When creating a project with maturin or scikit-build-core, fyn configures tool.fyn.cache-keys to include common source file types. To force a rebuild, e.g. when changing files outside cache-keys or when not using cache-keys, use --reinstall.

Creating a minimal project

If you only want to create a pyproject.toml, use the --bare option:

$ fyn init example --bare

fyn will skip creating a Python version pin file, a README, and any source directories or files. Additionally, fyn will not initialize a version control system (i.e., git).

$ tree example-bare
example-bare
└── pyproject.toml

fyn will also not add extra metadata to the pyproject.toml, such as the description or authors.

[project]
name = "example"
version = "0.1.0"
requires-python = ">=3.12"
dependencies = []

The --bare option can be used with other options like --lib or --build-backend — in these cases fyn will still configure a build system but will not create the expected file structure.

When --bare is used, additional features can still be used opt-in:

$ fyn init example --bare --description "Hello world" --author-from git --vcs git --python-pin