{"id":13493813,"url":"https://github.com/23andMe/Yamale","last_synced_at":"2025-03-28T12:32:27.892Z","repository":{"id":13582003,"uuid":"16274633","full_name":"23andMe/Yamale","owner":"23andMe","description":"A schema and validator for YAML.","archived":false,"fork":false,"pushed_at":"2024-07-01T18:15:56.000Z","size":827,"stargazers_count":680,"open_issues_count":44,"forks_count":88,"subscribers_count":56,"default_branch":"master","last_synced_at":"2024-10-29T14:54:23.776Z","etag":null,"topics":["python","schema","yaml"],"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/23andMe.png","metadata":{"files":{"readme":"README.md","changelog":null,"contributing":null,"funding":null,"license":"LICENSE","code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":"SECURITY.md","support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null}},"created_at":"2014-01-27T09:30:25.000Z","updated_at":"2024-10-29T04:58:19.000Z","dependencies_parsed_at":"2023-01-13T17:32:47.988Z","dependency_job_id":"75f46064-4a03-4202-9f47-a459d913c5be","html_url":"https://github.com/23andMe/Yamale","commit_stats":{"total_commits":250,"total_committers":44,"mean_commits":5.681818181818182,"dds":0.552,"last_synced_commit":"bacaa1d8e20395e11fe087cb7a7cb0365c2afd50"},"previous_names":[],"tags_count":61,"template":false,"template_full_name":null,"repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/23andMe%2FYamale","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/23andMe%2FYamale/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/23andMe%2FYamale/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/23andMe%2FYamale/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/23andMe","download_url":"https://codeload.github.com/23andMe/Yamale/tar.gz/refs/heads/master","host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":245767347,"owners_count":20668825,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2022-07-04T15:15:14.044Z","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":["python","schema","yaml"],"created_at":"2024-07-31T19:01:19.082Z","updated_at":"2025-03-28T12:32:27.877Z","avatar_url":"https://github.com/23andMe.png","language":"Python","funding_links":[],"categories":["Python","yaml"],"sub_categories":[],"readme":"Yamale (ya·ma·lē)\n=================\n[![Build Status](https://github.com/23andMe/Yamale/actions/workflows/run-tests.yml/badge.svg)](https://github.com/23andMe/Yamale/actions/workflows/run-tests.yml)\n[![PyPI](https://img.shields.io/pypi/v/yamale.svg)](https://pypi.python.org/pypi/yamale)\n[![downloads](https://static.pepy.tech/badge/yamale/month)](https://pepy.tech/project/yamale)\n[![versions](https://img.shields.io/pypi/pyversions/yamale.svg)](https://github.com/yamale/yamale)\n[![license](https://img.shields.io/github/license/23andMe/yamale.svg)](https://github.com/23andMe/Yamale/blob/master/LICENSE)\n\n| :warning: Ensure that your schema definitions come from internal or trusted sources. Yamale does not protect against intentionally malicious schemas. |\n|:------------|\n\n\u003cimg src=\"https://github.com/23andMe/Yamale/blob/master/yamale.png?raw=true\" alt=\"Yamale\" width=\"400\"/\u003e\n\nA schema and validator for YAML.\n\nWhat's YAML? See the current spec [here](http://www.yaml.org/spec/1.2/spec.html) and an introduction\nto the syntax [here](https://github.com/Animosity/CraftIRC/wiki/Complete-idiot's-introduction-to-yaml).\n\n\nRequirements\n------------\n* Python 3.8+\n* PyYAML\n* ruamel.yaml (optional)\n\nInstall\n-------\n### pip\n```\n$ pip install yamale\n# or to include ruamel.yaml as a dependency\n$ pip install yamale[ruamel]\n```\n\nNOTE: Some platforms, e.g., Mac OS, may ship with only Python 2 and may not have pip installed.\nInstallation of Python 3 should also install pip. To preserve any system dependencies on default\nsoftware, consider installing Python 3 as a local package. Please note replacing system-provided\nPython may disrupt other software. Mac OS users may wish to investigate MacPorts, homebrew, or\nbuilding Python 3 from source; in all three cases, Apple's Command Line Tools (CLT) for Xcode\nmay be required. See also [developers](#developers), below.\n\n### Manual\n1. Download Yamale from: https://github.com/23andMe/Yamale/archive/master.zip\n2. Unzip somewhere temporary\n3. Run `python setup.py install` (may have to prepend `sudo`)\n\nUsage\n-----\n### Command line\nYamale can be run from the command line to validate one or many YAML files. Yamale will search the\ndirectory you supply (current directory is default) for YAML files. Each YAML file it finds it will\nlook in the same directory as that file for its schema, if there is no schema Yamale will keep\nlooking up the directory tree until it finds one. If Yamale can not find a schema it will tell you.\n\nUsage:\n\n```\nusage: yamale [-h] [-s SCHEMA] [-e PATTERN] [-p PARSER] [-n CPU_NUM] [-x] [-v] [-V] [PATH ...]\n\nValidate yaml files.\n\npositional arguments:\n  PATH                  Paths to validate, either directories or files. Default is the current directory.\n\noptions:\n  -h, --help            show this help message and exit\n  -s SCHEMA, --schema SCHEMA\n                        filename of schema. Default is schema.yaml.\n  -e PATTERN, --exclude PATTERN\n                        Python regex used to exclude files from validation. Any substring match of a file's absolute path will be excluded. Uses\n                        default Python3 regex. Option can be supplied multiple times.\n  -p PARSER, --parser PARSER\n                        YAML library to load files. Choices are \"ruamel\" or \"pyyaml\" (default).\n  -n CPU_NUM, --cpu-num CPU_NUM\n                        Number of child processes to spawn for validation. Default is 4. 'auto' to use CPU count.\n  -x, --no-strict       Disable strict mode, unexpected elements in the data will be accepted.\n  -v, --verbose         show verbose information\n  -V, --version         show program's version number and exit\n```\n\n### API\nThere are several ways to feed Yamale schema and data files. The simplest way is to let Yamale take\ncare of reading and parsing your YAML files.\n\nAll you need to do is supply the files' path:\n```python\n# Import Yamale and make a schema object:\nimport yamale\nschema = yamale.make_schema('./schema.yaml')\n\n# Create a Data object\ndata = yamale.make_data('./data.yaml')\n\n# Validate data against the schema. Throws a ValueError if data is invalid.\nyamale.validate(schema, data)\n```\n\nYou can pass a string of YAML to `make_schema()` and `make_data()` instead of passing a file path\nby using the `content=` parameter:\n\n```python\ndata = yamale.make_data(content=\"\"\"\nname: Bill\nage: 26\nheight: 6.2\nawesome: True\n\"\"\")\n```\n\nIf `data` is valid, nothing will happen. However, if `data` is invalid Yamale will throw a\n`YamaleError` with a message containing all the invalid nodes:\n```python\ntry:\n    yamale.validate(schema, data)\n    print('Validation success! 👍')\nexcept ValueError as e:\n    print('Validation failed!\\n%s' % str(e))\n    exit(1)\n```\nand an array of `ValidationResult`.\n```python\ntry:\n    yamale.validate(schema, data)\n    print('Validation success! 👍')\nexcept YamaleError as e:\n    print('Validation failed!\\n')\n    for result in e.results:\n        print(\"Error validating data '%s' with '%s'\\n\\t\" % (result.data, result.schema))\n        for error in result.errors:\n            print('\\t%s' % error)\n    exit(1)\n```\n\nYou can also specify an optional `parser` if you'd like to use the `ruamel.yaml` (YAML 1.2 support) instead:\n```python\n# Import Yamale and make a schema object, make sure ruamel.yaml is installed already.\nimport yamale\nschema = yamale.make_schema('./schema.yaml', parser='ruamel')\n\n# Create a Data object\ndata = yamale.make_data('./data.yaml', parser='ruamel')\n\n# Validate data against the schema same as before.\nyamale.validate(schema, data)\n```\n\n### Schema\n\n| :warning: Ensure that your schema definitions come from internal or trusted sources. Yamale does not protect against intentionally malicious schemas. |\n|:------------|\n\nTo use Yamale you must make a schema. A schema is a valid YAML file with one or more documents\ninside. Each node terminates in a string which contains valid Yamale syntax. For example, `str()`\nrepresents a [String validator](#validators).\n\nA basic schema:\n```yaml\nname: str()\nage: int(max=200)\nheight: num()\nawesome: bool()\n```\n\nAnd some YAML that validates:\n```yaml\nname: Bill\nage: 26\nheight: 6.2\nawesome: True\n```\n\nTake a look at the [Examples](#examples) section for more complex schema ideas.\n\n#### Includes\nSchema files may contain more than one YAML document (nodes separated by `---`). The first document\nfound will be the base schema. Any additional documents will be treated as Includes. Includes allow\nyou to define a valid structure once and use it several times. They also allow you to do recursion.\n\nA schema with an Include validator:\n```yaml\nperson1: include('person')\nperson2: include('person')\n---\nperson:\n    name: str()\n    age: int()\n```\n\nSome valid YAML:\n```yaml\nperson1:\n    name: Bill\n    age: 70\n\nperson2:\n    name: Jill\n    age: 20\n```\n\nEvery root node not in the first YAML document will be treated like an include:\n```yaml\nperson: include('friend')\ngroup: include('family')\n---\nfriend:\n    name: str()\nfamily:\n    name: str()\n```\n\nIs equivalent to:\n```yaml\nperson: include('friend')\ngroup: include('family')\n---\nfriend:\n    name: str()\n---\nfamily:\n    name: str()\n```\n\n##### Recursion\nYou can get recursion using the Include validator.\n\nThis schema:\n```yaml\nperson: include('human')\n---\nhuman:\n    name: str()\n    age: int()\n    friend: include('human', required=False)\n```\n\nWill validate this data:\n```yaml\nperson:\n    name: Bill\n    age: 50\n    friend:\n        name: Jill\n        age: 20\n        friend:\n            name: Will\n            age: 10\n```\n\n##### Adding external includes\nAfter you construct a schema you can add extra, external include definitions by calling\n`schema.add_include(dict)`. This method takes a dictionary and adds each key as another include.\n\n### Strict mode\nBy default Yamale will provide errors for extra elements present in lists and maps that are not\ncovered by the schema. With strict mode disabled (using the `--no-strict` command line option),\nadditional elements will not cause any errors. In the API, strict mode can be toggled by passing\nthe strict=True/False flag to the validate function.\n\nIt is possible to mix strict and non-strict mode by setting the strict=True/False flag in the\ninclude validator, setting the option only for the included validators.\n\nValidators\n----------\nHere are all the validators Yamale knows about. Every validator takes a `required` keyword telling\nYamale whether or not that node must exist. By default every node is required. Example: `str(required=False)`\n\nYou can also require that an optional value is not `None` by using the `none` keyword. By default\nYamale will accept `None` as a valid value for a key that's not required. Reject `None` values\nwith `none=False` in any validator. Example: `str(required=False, none=False)`.\n\nSome validators take keywords and some take arguments, some take both. For instance the `enum()`\nvalidator takes one or more constants as arguments and the `required` keyword:\n`enum('a string', 1, False, required=False)`\n\n### String - `str(min=int, max=int, equals=string, starts_with=string, ends_with=string, matches=regex, exclude=string, ignore_case=False, multiline=False, dotall=False)`\nValidates strings.\n- keywords\n    - `min`: len(string) \u003e= min\n    - `max`: len(string) \u003c= max\n    - `equals`: string == value (add `ignore_case=True` for case-insensitive checking)\n    - `starts_with`: Accepts only strings starting with given value (add `ignore_case=True` for\n      case-insensitive checking)\n    - `matches`: Validates the string against a given regex. Similar to the `regex()` validator,\n      you can use `ignore_case`, `multiline` and `dotall`)\n    - `ends_with`: Accepts only strings ending with given value (add `ignore_case=True` for case-insensitive checking)\n    - `exclude`: Rejects strings that contains any character in the excluded value\n    - `ignore_case`: Validates strings in a case-insensitive manner.\n    - `multiline`: `^` and `$` in a pattern match at the beginning and end of each line in a string\n       in addition to matching at the beginning and end of the entire string. (A pattern matches\n       at [the beginning of a string](https://docs.python.org/3/library/re.html#re.match) even in\n       multiline mode; see below for a workaround.); only allowed in conjunction with a `matches` keyword.\n    - `dotall`: `.` in a pattern matches newline characters in a validated string in addition to\n      matching every character that *isn't* a newline.; only allowed in conjunction with a `matches` keyword.\n\nExamples:\n- `str(max=10, exclude='?!')`: Allows only strings less than 11 characters that don't contain `?` or `!`.\n\n### Regex - `regex([patterns], name=string, ignore_case=False, multiline=False, dotall=False)`\nValidates strings against one or more regular expressions.\n- arguments: one or more Python regular expression patterns\n- keywords:\n    - `name`: A friendly description for the patterns.\n    - `ignore_case`: Validates strings in a case-insensitive manner.\n    - `multiline`: `^` and `$` in a pattern match at the beginning and end of each line in a string\n       in addition to matching at the beginning and end of the entire string. (A pattern matches\n       at [the beginning of a string](https://docs.python.org/3/library/re.html#re.match) even in\n       multiline mode; see below for a workaround.)\n    - `dotall`: `.` in a pattern matches newline characters in a validated string in addition to\n      matching every character that *isn't* a newline.\n\nExamples:\n- `regex('^[^?!]{,10}$')`: Allows only strings less than 11 characters that don't contain `?` or `!`.\n- `regex(r'^(\\d+)(\\s\\1)+$', name='repeated natural')`: Allows only strings that contain two or\n  more identical digit sequences, each separated by a whitespace character. Non-matching strings\n  like `sugar` are rejected with a message like `'sugar' is not a repeated natural.`\n- `regex('.*^apples$', multiline=True, dotall=True)`: Allows the string `apples` as well\n  as multiline strings that contain the line `apples`.\n\n### Integer - `int(min=int, max=int)`\nValidates integers.\n- keywords\n    - `min`: int \u003e= min\n    - `max`: int \u003c= max\n\n### Number - `num(min=float, max=float)`\nValidates integers and floats.\n- keywords\n    - `min`: num \u003e= min\n    - `max`: num \u003c= max\n\n### Boolean - `bool()`\nValidates booleans.\n\n### Null - `null()`\nValidates null values.\n\n### Enum - `enum([primitives])`\nValidates from a list of constants.\n- arguments: constants to test equality with\n\nExamples:\n- `enum('a string', 1, False)`: a value can be either `'a string'`, `1` or `False`\n\n### Day - `day(min=date, max=date)`\nValidates a date in the form of YYYY-MM-DD.\n- keywords\n    - `min`: date \u003e= min\n    - `max`: date \u003c= max\n\nExamples:\n- `day(min='2001-01-01', max='2100-01-01')`: Only allows dates between 2001-01-01 and 2100-01-01.\n\n### Timestamp - `timestamp(min=time, max=time)`\nValidates a timestamp in the form of YYYY-MM-DD HH:MM:SS.\n- keywords\n    - `min`: time \u003e= min\n    - `max`: time \u003c= max\n\nExamples:\n- `timestamp(min='2001-01-01 01:00:00', max='2100-01-01 23:00:00')`: Only allows times between\n  2001-01-01 01:00:00 and 2100-01-01 23:00:00.\n\n### List - `list([validators], min=int, max=int)`\nValidates lists. If one or more validators are passed to `list()` only nodes that pass at\nleast one of those validators will be accepted.\n\n- arguments: one or more validators to test values with\n- keywords\n    - `min`: len(list) \u003e= min\n    - `max`: len(list) \u003c= max\n\nExamples:\n- `list()`: Validates any list\n- `list(include('custom'), int(), min=4)`: Only validates lists that contain the `custom` include\n  or integers and contains a minimum of 4 items.\n\n### Map - `map([validators], key=validator, min=int, max=int)`\nValidates maps. Use when you want a node to contain freeform data. Similar to `List`, `Map` takes\none or more validators to run against the values of its nodes, and only nodes that pass at least\none of those validators will be accepted. By default, only the values of nodes are validated and\nthe keys aren't checked.\n- arguments: one or more validators to test values with\n- keywords\n    - `key`: A validator for the keys of the map.\n    - `min`: len(map) \u003e= min\n    - `max`: len(map) \u003c= max\n\nExamples:\n- `map()`: Validates any map\n- `map(str(), int())`: Only validates maps whose values are strings or integers.\n- `map(str(), key=int())`: Only validates maps whose keys are integers and values are strings. `1: one` would be valid but `'1': one` would not.\n- `map(str(), min=1)`: Only validates a non-empty map.\n\n### IP Address - `ip()`\nValidates IPv4 and IPv6 addresses.\n\n- keywords\n    - `version`: 4 or 6; explicitly force IPv4 or IPv6 validation\n\nExamples:\n- `ip()`: Allows any valid IPv4 or IPv6 address\n- `ip(version=4)`: Allows any valid IPv4 address\n- `ip(version=6)`: Allows any valid IPv6 address\n\n### MAC Address - `mac()`\nValidates MAC addresses.\n\nExamples:\n- `mac()`: Allows any valid MAC address\n\n### SemVer (Semantic Versioning) - `semver()`\nValidates [Semantic Versioning](https://semver.org/) strings.\n\nExamples:\n- `semver()`: Allows any valid SemVer string\n\n### Any - `any([validators])`\nValidates against a union of types. Use when a node **must** contain **one and only one** of several types. It is valid\nif at least one of the listed validators is valid. If no validators are given, accept any value.\n- arguments: validators to test values with (if none is given, allow any value; if one or more are given,\none must be present)\n\nExamples:\n- `any(int(), null())`: Validates either an integer **or** a null value.\n- `any(num(), include('vector'))`: Validates **either** a number **or** an included 'vector' type.\n- `any(str(min=3, max=3),str(min=5, max=5),str(min=7, max=7))`: validates to a string that is exactly 3, 5, or 7 characters long\n- `any()`: Allows any value.\n\n### Subset - `subset([validators], allow_empty=False)`\nValidates against a subset of types. Unlike the `Any` validator, this validators allows **one or more** of several types.\nAs such, it *automatically validates against a list*. It is valid if all values can be validated against at least one\nvalidator.\n- arguments: validators to test with (at least one; if none is given, a `ValueError` exception will be raised)\n- keywords:\n    - `allow_empty`: Allow the subset to be empty (and is, therefore, also optional). This overrides the `required`\nflag.\n\nExamples:\n- `subset(int(), str())`: Validators against an integer, a string, or a list of either.\n- `subset(int(), str(), allow_empty=True)`: Same as above, but allows the empty set and makes the subset optional.\n\n### Include - `include(include_name)`\nValidates included structures. Must supply the name of a valid include.\n- arguments: single name of a defined include, surrounded by quotes.\n\nExamples:\n- `include('person')`\n\n### Custom validators\nIt is also possible to add your own custom validators. This is an advanced topic, but here is an\nexample of adding a `Date` validator and using it in a schema as `date()`\n\n```python\nimport yamale\nimport datetime\nfrom yamale.validators import DefaultValidators, Validator\n\nclass Date(Validator):\n    \"\"\" Custom Date validator \"\"\"\n    tag = 'date'\n\n    def _is_valid(self, value):\n        return isinstance(value, datetime.date)\n\nvalidators = DefaultValidators.copy()  # This is a dictionary\nvalidators[Date.tag] = Date\nschema = yamale.make_schema('./schema.yaml', validators=validators)\n# Then use `schema` as normal\n```\n\nExamples\n--------\n\n| :warning: Ensure that your schema definitions come from internal or trusted sources. Yamale does not protect against intentionally malicious schemas. |\n|:------------|\n\n### Using keywords\n#### Schema:\n```yaml\noptional: str(required=False)\noptional_min: int(min=1, required=False)\nmin: num(min=1.5)\nmax: int(max=100)\n```\n#### Valid Data:\n```yaml\noptional_min: 10\nmin: 1.6\nmax: 100\n```\n\n### Includes and recursion\n#### Schema:\n```yaml\ncustomerA: include('customer')\ncustomerB: include('customer')\nrecursion: include('recurse')\n---\ncustomer:\n    name: str()\n    age: int()\n    custom: include('custom_type')\n\ncustom_type:\n    integer: int()\n\nrecurse:\n    level: int()\n    again: include('recurse', required=False)\n```\n#### Valid Data:\n```yaml\ncustomerA:\n    name: bob\n    age: 900\n    custom:\n        integer: 1\ncustomerB:\n    name: jill\n    age: 1\n    custom:\n        integer: 3\nrecursion:\n    level: 1\n    again:\n        level: 2\n        again:\n            level: 3\n            again:\n                level: 4\n```\n\n### Lists\n#### Schema:\n```yaml\nlist_with_two_types: list(str(), include('variant'))\nquestions: list(include('question'))\n---\nvariant:\n  rsid: str()\n  name: str()\n\nquestion:\n  choices: list(include('choices'))\n  questions: list(include('question'), required=False)\n\nchoices:\n  id: str()\n```\n#### Valid Data:\n```yaml\nlist_with_two_types:\n  - 'some'\n  - rsid: 'rs123'\n    name: 'some SNP'\n  - 'thing'\n  - rsid: 'rs312'\n    name: 'another SNP'\nquestions:\n  - choices:\n      - id: 'id_str'\n      - id: 'id_str1'\n    questions:\n      - choices:\n        - id: 'id_str'\n        - id: 'id_str1'\n```\n\n### The data is a list of items without a keyword at the top level\n#### Schema:\n```yaml\nlist(include('human'), min=2, max=2)\n\n---\nhuman:\n  name: str()\n  age: int(max=200)\n  height: num()\n  awesome: bool()\n```\n#### Valid Data:\n```yaml\n- name: Bill\n  age: 26\n  height: 6.2\n  awesome: True\n\n- name: Adrian\n  age: 23\n  height: 6.3\n  awesome: True\n```\n\nWriting Tests\n-------------\nTo validate YAML files when you run your program's tests use Yamale's YamaleTestCase\n\nExample:\n\n```python\nclass TestYaml(YamaleTestCase):\n    base_dir = os.path.dirname(os.path.realpath(__file__))\n    schema = 'schema.yaml'\n    yaml = 'data.yaml'\n    # or yaml = ['data-*.yaml', 'some_data.yaml']\n\n    def runTest(self):\n        self.assertTrue(self.validate())\n```\n\n`base_dir`: String path to prepend to all other paths. This is optional.\n\n`schema`: String of path to the schema file to use. One schema file per test case.\n\n`yaml`: String or list of yaml files to validate. Accepts globs.\n\nDevelopers\n----------\n### Linting + Formatting\nYamale is formatted with [ruff](https://github.com/astral-sh/ruff). There is a github action enforcing\nruff formatting and linting rules. You can run this locally via `make lint` or by installing\nthe pre-commit hooks via `make install-hooks`\n\n### Testing\nYamale uses [Tox](https://tox.readthedocs.org/en/latest/) to run its tests against multiple Python\nversions. To run tests, first checkout Yamale, install Tox, then run `make test` in Yamale's root\ndirectory. You may also have to install the correct Python versions to test with as well.\n\nNOTE on Python versions: `tox.ini` specifies the lowest and highest versions of Python supported by\nYamale. Unless your development environment is configured to support testing against multiple Python\nversions, one or more of the test branches may fail. One method of enabling testing against multiple\nversions of Python is to install `pyenv` and `tox-pyenv` and to use `pyenv install` and `pyenv local`\nto ensure that tox is able to locate appropriate Pythons.\n\n### Releasing\nYamale uses Github Actions to upload new tags to PyPi.\nTo release a new version:\n\n1. Make a commit with the new version number in `yamale/VERSION`.\n1. Run tests for good luck.\n1. Run `make release`.\n\nGithub Actions will take care of the rest.\n","project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2F23andMe%2FYamale","html_url":"https://awesome.ecosyste.ms/projects/github.com%2F23andMe%2FYamale","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2F23andMe%2FYamale/lists"}