{"id":32175950,"url":"https://github.com/robotools/fontparts","last_synced_at":"2026-05-26T15:01:44.097Z","repository":{"id":5356326,"uuid":"52969280","full_name":"robotools/fontParts","owner":"robotools","description":"The replacement for RoboFab","archived":false,"fork":false,"pushed_at":"2026-05-25T18:31:59.000Z","size":2769,"stargazers_count":148,"open_issues_count":105,"forks_count":45,"subscribers_count":28,"default_branch":"master","last_synced_at":"2026-05-25T18:32:35.037Z","etag":null,"topics":["font","python","scripting"],"latest_commit_sha":null,"homepage":"","language":"Python","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/robotools.png","metadata":{"files":{"readme":"README.rst","changelog":"NEWS.rst","contributing":"CONTRIBUTING.rst","funding":null,"license":"LICENSE","code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null,"zenodo":null,"notice":null,"maintainers":null,"copyright":null,"agents":null,"dco":null,"cla":null}},"created_at":"2016-03-02T14:30:03.000Z","updated_at":"2026-05-15T13:54:34.000Z","dependencies_parsed_at":"2024-01-11T14:08:36.084Z","dependency_job_id":"4368ca8d-de34-4564-a846-f4367b3a717f","html_url":"https://github.com/robotools/fontParts","commit_stats":{"total_commits":1436,"total_committers":35,"mean_commits":41.02857142857143,"dds":0.692200557103064,"last_synced_commit":"f5250dc93916f45d5bf3e286cf44ff4c7dad9965"},"previous_names":[],"tags_count":68,"template":false,"template_full_name":null,"purl":"pkg:github/robotools/fontParts","repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/robotools%2FfontParts","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/robotools%2FfontParts/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/robotools%2FfontParts/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/robotools%2FfontParts/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/robotools","download_url":"https://codeload.github.com/robotools/fontParts/tar.gz/refs/heads/master","sbom_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/robotools%2FfontParts/sbom","scorecard":{"id":781208,"data":{"date":"2025-08-11","repo":{"name":"github.com/robotools/fontParts","commit":"2aa2139fedce84a94182ae1bf61cab96de87d623"},"scorecard":{"version":"v5.2.1-40-gf6ed084d","commit":"f6ed084d17c9236477efd66e5b258b9d4cc7b389"},"score":5,"checks":[{"name":"Code-Review","score":3,"reason":"Found 9/26 approved changesets -- score normalized to 3","details":null,"documentation":{"short":"Determines if the project requires human code review before pull requests (aka merge requests) are merged.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#code-review"}},{"name":"Packaging","score":-1,"reason":"packaging workflow not detected","details":["Warn: no GitHub/GitLab publishing workflow detected."],"documentation":{"short":"Determines if the project is published as a package that others can easily download, install, easily update, and uninstall.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#packaging"}},{"name":"Token-Permissions","score":0,"reason":"detected GitHub workflow tokens with excessive permissions","details":["Warn: no topLevel permission defined: .github/workflows/publish-package.yml:1","Warn: no topLevel permission defined: .github/workflows/run-tests.yml:1","Info: no jobLevel write permissions found"],"documentation":{"short":"Determines if the project's workflows follow the principle of least privilege.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#token-permissions"}},{"name":"Dangerous-Workflow","score":10,"reason":"no dangerous workflow patterns detected","details":null,"documentation":{"short":"Determines if the project's GitHub Action workflows avoid dangerous patterns.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#dangerous-workflow"}},{"name":"Maintained","score":10,"reason":"22 commit(s) and 7 issue activity found in the last 90 days -- score normalized to 10","details":null,"documentation":{"short":"Determines if the project is \"actively maintained\".","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#maintained"}},{"name":"CII-Best-Practices","score":0,"reason":"no effort to earn an OpenSSF best practices badge detected","details":null,"documentation":{"short":"Determines if the project has an OpenSSF (formerly CII) Best Practices Badge.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#cii-best-practices"}},{"name":"Binary-Artifacts","score":10,"reason":"no binaries found in the repo","details":null,"documentation":{"short":"Determines if the project has generated executable (binary) artifacts in the source repository.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#binary-artifacts"}},{"name":"Pinned-Dependencies","score":0,"reason":"dependency not pinned by hash detected -- score normalized to 0","details":["Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/publish-package.yml:14: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/publish-package.yml/master?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/publish-package.yml:28: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/publish-package.yml/master?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/publish-package.yml:30: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/publish-package.yml/master?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/run-tests.yml:37: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/run-tests.yml/master?enable=pin","Warn: GitHub-owned GitHubAction not pinned by hash: .github/workflows/run-tests.yml:39: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/run-tests.yml/master?enable=pin","Warn: third-party GitHubAction not pinned by hash: .github/workflows/run-tests.yml:51: update your workflow using https://app.stepsecurity.io/secureworkflow/robotools/fontParts/run-tests.yml/master?enable=pin","Warn: pipCommand not pinned by hash: .github/workflows/publish-package.yml:35","Warn: pipCommand not pinned by hash: .github/workflows/publish-package.yml:36","Warn: pipCommand not pinned by hash: .github/workflows/run-tests.yml:44","Info:   0 out of   5 GitHub-owned GitHubAction dependencies pinned","Info:   0 out of   1 third-party GitHubAction dependencies pinned","Info:   0 out of   3 pipCommand dependencies pinned"],"documentation":{"short":"Determines if the project has declared and pinned the dependencies of its build process.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#pinned-dependencies"}},{"name":"Security-Policy","score":0,"reason":"security policy file not detected","details":["Warn: no security policy file detected","Warn: no security file to analyze","Warn: no security file to analyze","Warn: no security file to analyze"],"documentation":{"short":"Determines if the project has published a security policy.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#security-policy"}},{"name":"License","score":10,"reason":"license file detected","details":["Info: project has a license file: LICENSE:0","Info: FSF or OSI recognized license: MIT License: LICENSE:0"],"documentation":{"short":"Determines if the project has defined a license.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#license"}},{"name":"Fuzzing","score":0,"reason":"project is not fuzzed","details":["Warn: no fuzzer integrations found"],"documentation":{"short":"Determines if the project uses fuzzing.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#fuzzing"}},{"name":"Signed-Releases","score":-1,"reason":"no releases found","details":null,"documentation":{"short":"Determines if the project cryptographically signs release artifacts.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#signed-releases"}},{"name":"Branch-Protection","score":-1,"reason":"internal error: error during branchesHandler.setup: internal error: githubv4.Query: Resource not accessible by integration","details":null,"documentation":{"short":"Determines if the default and release branches are protected with GitHub's branch protection settings.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#branch-protection"}},{"name":"Vulnerabilities","score":9,"reason":"1 existing vulnerabilities detected","details":["Warn: Project is vulnerable to: PYSEC-2024-187 / GHSA-rqc4-2hc7-8c8v"],"documentation":{"short":"Determines if the project has open, known unfixed vulnerabilities.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#vulnerabilities"}},{"name":"SAST","score":0,"reason":"SAST tool is not run on all commits -- score normalized to 0","details":["Warn: 0 commits out of 19 are checked with a SAST tool"],"documentation":{"short":"Determines if the project uses static code analysis.","url":"https://github.com/ossf/scorecard/blob/f6ed084d17c9236477efd66e5b258b9d4cc7b389/docs/checks.md#sast"}}]},"last_synced_at":"2025-08-23T04:57:42.262Z","repository_id":5356326,"created_at":"2025-08-23T04:57:42.262Z","updated_at":"2025-08-23T04:57:42.262Z"},"host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":286080680,"owners_count":33525947,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2026-05-26T03:12:49.672Z","status":"ssl_error","status_checked_at":"2026-05-26T03:12:47.976Z","response_time":63,"last_error":"SSL_read: unexpected eof while reading","robots_txt_status":"success","robots_txt_updated_at":"2025-07-24T06:49:26.215Z","robots_txt_url":"https://github.com/robots.txt","online":false,"can_crawl_api":true,"host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["font","python","scripting"],"created_at":"2025-10-21T19:52:23.695Z","updated_at":"2026-05-26T15:01:44.053Z","avatar_url":"https://github.com/robotools.png","language":"Python","funding_links":[],"categories":[],"sub_categories":[],"readme":"|CI Build Status| |Coverage| |PyPI| |Versions|\n\nFontParts\n~~~~~~~~~\n\nAn API for interacting with the parts of fonts during the font\ndevelopment process. FontParts is the replacement for\n`RoboFab \u003chttp://robofab.com\u003e`__. The project has a\n`MIT open-source licence \u003cLICENSE\u003e`__.\n\nThe documentation is at\n`fontparts.readthedocs.io \u003chttp://fontparts.readthedocs.io/en/latest/\u003e`__.\n\n*This is a work in progress. We are still working out the API, abstract\nimplementation, example implementation, test suite and documentation.*\n\nWant to contribute?\n-------------------\n\nThank you! Please see the `CONTRIBUTING.rst \u003chttps://github.com/robofab-developers/fontParts/blob/master/CONTRIBUTING.rst\u003e`_ file for a guide on how to help.\n\nAlso, feedback is very much welcome, please open an issue when you run\ninto something that you wish fontParts did/didn't do.\n\n\nInstallation\n~~~~~~~~~~~~\n\nFontParts requires `Python \u003chttp://www.python.org/download/\u003e`__ 3.8 or later.\n\nThe package is listed in the Python Package Index (PyPI), so you can\ninstall it with `pip \u003chttps://pip.pypa.io\u003e`__:\n\n.. code:: sh\n\n    pip install fontParts\n\nIf you would like to contribute to its development, you can clone the\nrepository from Github, install the package in 'editable' mode and\nmodify the source code in place. We recommend creating a virtual\nenvironment, using `virtualenv \u003chttps://virtualenv.pypa.io\u003e`__ or `venv \u003chttps://docs.python.org/3/library/venv.html\u003e`__ module.\n\n.. code:: sh\n\n    # download the source code to 'fontParts' folder\n    git clone https://github.com/robofab-developers/fontParts.git\n    cd fontParts\n\n    # create new virtual environment called e.g. 'fontParts-venv', or anything you like\n    python -m virtualenv fontParts-venv\n\n    # source the `activate` shell script to enter the environment (Un\\*x); to exit, just type `deactivate`\n    . fontParts-venv/bin/activate\n\n    # to activate the virtual environment in Windows `cmd.exe`, do\n    fontParts-venv\\Scripts\\activate.bat\n\n    # install in 'editable' mode\n    pip install -e .\n\nRoadmap\n~~~~~~~\n\nWe are currently working towards the 1.0 release.\n\n* **0.8** Initial releases. Python 2 \u0026 3.\n* **0.9** Python 3 only.\n* **1.0** Documentation and testing complete.\n* **1.5** Removal of ``Deprecated``. Released 1 year after 1.0.\n\nTesting\n~~~~~~~\n\nTesting is setup so that each environment that includes fontParts\ncan provides the objects needed to run a common set of tests.\nThis makes testing very easy for environments that use fontParts (for\nexample, see the fontshell\n`test.py \u003chttps://github.com/robofab-developers/fontParts/blob/master/Lib/fontParts/fontshell/test.py\u003e`__\nscript), but it means testing is different than other python packages.\n\nAutomated testing of the package is done in the fontshell environment.\nfontshell is fontParts for the commandline, implemented with\n`defcon \u003chttps://github.com/typesupply/defcon\u003e`__ and is included\nas part of the fontParts package.\n\nBefore you can run the test suite you’ll need to install the test dependencies:\n\n.. code:: sh\n\n    pip install -r requirements-dev.txt\n\nTo run the test suite you can do:\n\n.. code:: sh\n\n    python Lib/fontParts/fontshell/test.py\n\nTo test in other environments, run the test script provided by that environment.\n\nYou can also use `tox \u003chttps://testrun.org/tox/latest/\u003e`__ to\nautomatically run tests on different Python versions in isolated virtual\nenvironments.\n\n.. code:: sh\n\n    pip install tox\n    tox\n\nNote that when you run ``tox`` without arguments, the tests are executed\nfor all the environments listed in tox.ini's ``envlist``. In our case,\nthis is Python 3.6, so for this to work the ``python3.6`` executables must\nbe available in your ``PATH``.\n\nYou can specify an alternative environment list via the ``-e`` option,\nor the ``TOXENV`` environment variable:\n\n.. code:: sh\n\n    tox -e py39-nocov\n    TOXENV=\"py36-cov,htmlcov\" tox\n\n.. |CI Build Status| image:: https://github.com/robotools/fontParts/workflows/Tests/badge.svg\n   :target: https://github.com/robotools/fontParts/actions?query=workflow%3ATests\n.. |PyPI| image:: https://img.shields.io/pypi/v/fontParts.svg\n   :target: https://pypi.org/project/fontParts\n.. |Versions| image:: https://img.shields.io/badge/python-3.8%2C%203.9%2C%203.10%2C%203.11-blue.svg\n   :alt: Python Versions\n.. |Coverage| image:: https://codecov.io/gh/robotools/fontParts/branch/master/graph/badge.svg\n   :target: https://codecov.io/gh/robotools/fontParts\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frobotools%2Ffontparts","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Frobotools%2Ffontparts","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frobotools%2Ffontparts/lists"}