Skip to main content

Fake FontLab Studio 5 for automated tests and external scripting

Project description

FakeLab

Test-driven development for FontLab Studio 5 Python macros and modules.

FakeLab is a FontLab Studio 5 replacement for testing Python code.

Everything is only implemented so far as to make FontLab objects importable outside of FontLab Studio 5, and run tests.

It is suggested to install FakeLab in a virtual environment so FontLab won't accidentally import the fake module when running the scripts in actual FontLab Studio 5. FakeLab will run in Python 3; you have to live with any incompatibilites between Python 2.7 and 3 that may occur in your scripts.

Loading and saving VFBs is supported when you have the vfb extra installed, which uses vfbLib to read and write VFBs.

The implementation of FakeLab is based on the invaluable Unofficial FontLab/Python API Reference, and from running scripts in FontLab Studio and checking what they do, apart from crashing the application.

Installation

When you have activated your virtual environment:

$ pip install -e .

FL is then importable outside of FontLab Studio:

from FL import fl, Font

# Make an empty font
f = Font()

# Add the font to the mock app
fl.Add(f)

# Close the font
fl.Close()

If your are running in a virtual environment, and need to make your FontLab modules importable, add a .pth file in your virtual environment's site-packages directory:

# fl5modules.pth
/Users/<yourname>/Code/FLMacros/System/Modules

uv

To use tox with uv for testing, install and run like this:

uv sync --group test
uv run tox

A word of caution

FakeLab is a work in progress, and only implemented as far as I needed it in my own scripts. I have to admit that the bulk of them still has no tests.

If you encounter a NotImplementedError while writing tests, this is where you can excel at helping to improve this project ;) I'll be happy to accept your pull requests. Alternatively, open an issue, buy me a coffee, and hope that my kids leave me alone so I can find some time to work on your issue.

Writing tests

Developing scripts without automated testing is really only for very small projects. To be sure of the outcomes of a module or script, you should always write tests. This is usually done using pytest.

Tests example

Let's assume you have a FontLab script to select glyphs containing components. If you have your own tools collection for FontLab, this script may consist of two parts: One script that is listed in FontLab's macro toolbar, and one Python module implementing the logic, which is called by the toolbar script.

Studio 5
+- Macros
   +- Selection
      +- Select Composites.py
   +- System
      +- Modules
         +- fakeLabDemo
            +- selection
               +- __init__.py
               +- composites.py

The Select Composites.py script looks like this:

#FLM: Select composites
# Studio 5/Macros/Selection/Select Composites.py
from fakeLabDemo.selection.composites import selectComposites
selectComposites(fl.font)

And the module:

# Studio 5/Macros/System/Modules/fakeLabDemo/selection/composites.py
from __future__ import absolute_import, division, print_function

from FL import fl


def getFontIndex(font):
    """
    Get the index of the supplied font.
    We must iterate through the open fonts and compare file names,
    because the == operator can not compare the font objects directly.
    (FL font objects get a different id() each time they are called)

    :param font: A title for the dialog.
    :type font:  :py:class:`FL.Font`
    """
    for i in range(len(fl)):
        cf = fl[i]
        if cf.file_name == font.file_name:
            if font.file_name is None:
                if (
                    cf.family_name == font.family_name
                    and cf.style_name == font.style_name
                ):
                    return (cf, i)
            else:
                return (cf, i)
    # Font was not found, probably there are no open fonts
    return (None, -1)


def setSelection(font, glyph_names):
    """
    Set glyphs from the glyph_names list as selected in the font window.
    """
    f, i = getFontIndex(font)
    if i > -1:
        fl.ifont = i
        fl.Unselect()
        for n in glyph_names:
            fl.Select(n)


def selectComposites(font):
    """
    Select composites in font.
    """
    setSelection(
        font,
        [
            glyph.name
            for glyph in font.glyphs
            if glyph.components
        ]
    )

How can we be sure this script does what it is supposed to do? For pytest, we add another parallel folder structure to the existing structure:

Studio 5
+- Macros
   +- Selection
      +- Select Composites.py
   +- System
      +- Modules
         +- fakeLabDemo
            +- selection
               +- __init__.py
               +- composites.py
         +- tests
            +- fakeLabDemo
               +- selection
                  +- composites_test.py

The file composites_test.py, which is named analogous to the module file it relates to, is where we will implement our tests:

# Studio 5/Macros/System/Modules/tests/fakeLabDemo/selection/composites_test.py
import pytest

from FL import fl, Component, Font, Glyph, Point
from fakeLabDemo.selection.composites import selectComposites


def test_selectComposites():
    # Construct a fake FontLab font object
    font = Font()
    g = Glyph(1)
    g.name = "A"
    g.width = 500
    g.unicode = 0x41
    font.glyphs.append(g)

    g = Glyph(1)
    g.name = "dieresis"
    g.width = 500
    g.unicode = 0xA8
    font.glyphs.append(g)

    g = Glyph(1)
    g.name = "Adieresis"
    g.width = 500
    g.unicode = 0xC4
    g.components.append(Component(0))
    g.components.append(Component(1, Point(0, 300)))
    font.glyphs.append(g)

    # Add the font to the FL object
    fl.Add(font)
    fl.UpdateFont()

    # Run our script to be tested on the font
    selectComposites(fl.font)

    # You could save the fake font to JSON instead of VFB.
    # fl.font.Save("test_composites.vfb.json")

    # Test if the correct glyphs have been selected
    assert fl.Selected(0) == 0
    assert fl.Selected(1) == 0
    assert fl.Selected(2) == 1

    # Close the fake font
    fl.Close()

As you see, you can use the objects just as you would inside FontLab.

You can open a font from an existing VFB if you have the vfb extra installed. If not, you have to construct a test font using the FL Python API.

Invoke the test script in a Terminal window while your virtual environment is active:

cd "Studio 5/Macros/System/Modules"
python -m pytest tests/fakeLabDemo/selection/composites_test.py

If everything works out, you will see some output like this:

============================ test session starts ===============================
platform darwin -- Python 3.8.7, pytest-6.0.1, py-1.9.0, pluggy-0.13.1
rootdir: /Users/jens/Code/fakelab
collected 1 item

tests/fakeLabDemo/selection/composites_test.py .                          [100%]

============================= 1 passed in 0.02s ================================

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

fakelab-0.1.6.tar.gz (141.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

fakelab-0.1.6-py3-none-any.whl (167.9 kB view details)

Uploaded Python 3

File details

Details for the file fakelab-0.1.6.tar.gz.

File metadata

  • Download URL: fakelab-0.1.6.tar.gz
  • Upload date:
  • Size: 141.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for fakelab-0.1.6.tar.gz
Algorithm Hash digest
SHA256 2f95396dd6710e1c70f58c08c12211007e0027f3a61d2e421dbe980520b45bdf
MD5 f2728d4def125b6d2530e3381cd8f58e
BLAKE2b-256 6dbdf1d353856fb5b02fac26a828a189718d876ad6bcf4973c4d7fe26de24a57

See more details on using hashes here.

Provenance

The following attestation bundles were made for fakelab-0.1.6.tar.gz:

Publisher: publish_to_pypi.yml on jenskutilek/fakelab

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file fakelab-0.1.6-py3-none-any.whl.

File metadata

  • Download URL: fakelab-0.1.6-py3-none-any.whl
  • Upload date:
  • Size: 167.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for fakelab-0.1.6-py3-none-any.whl
Algorithm Hash digest
SHA256 983b514cc68ee36aa79bbdd7446303c016ecdc40a047e4711f06b822e27ee6d6
MD5 050a19f276a8317d86f8f9cd7ec428c4
BLAKE2b-256 d7314b4d1b1f6775810d536f31a548523a34ca01cf0a8080ab12a0c09ba23448

See more details on using hashes here.

Provenance

The following attestation bundles were made for fakelab-0.1.6-py3-none-any.whl:

Publisher: publish_to_pypi.yml on jenskutilek/fakelab

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page