Compare commits

..

No commits in common. "develop" and "0.1.0a0" have entirely different histories.

3788 changed files with 1393 additions and 11685 deletions

View file

@ -1,104 +1,3 @@
## 0.1.0a15 (2025-11-21)
### Fix
- ExtentionError => ExtensionError
## 0.1.0a14 (2025-11-06)
### Fix
- add translation
- update tests
## 0.1.0a13 (2025-10-29)
### Feat
- can define a new separator
## 0.1.0a12 (2025-10-10)
### Fix
- update test
- translation + tests
- tests for formatter
## 0.1.0a11 (2025-09-29)
### Feat
- default value for a calculated variable with an unknown optional variable
- update tests for integer type
## 0.1.0a10 (2025-09-22)
### Fix
- dictionary => structure
## 0.1.0a9 (2025-06-18)
### Fix
- rougail separation
## 0.1.0a8 (2025-05-12)
### Feat
- display loaded_from informations
### Fix
- black
## 0.1.0a7 (2025-05-02)
### Fix
- do not force use_data usage
## 0.1.0a6 (2025-04-30)
### Fix
- update tests
## 0.1.0a5 (2025-04-09)
### Fix
- version
## 0.1.0a4 (2025-04-01)
### Fix
- update tests
## 0.1.0a3 (2025-03-30)
### Fix
- update tests
- update tests, some errors are now in warnings level
## 0.1.0a2 (2025-02-10)
### Fix
- rename source
- add test without namespace
- update tests
## 0.1.0a1 (2024-11-27)
### Fix
- NAMESPACE. instead of NAMESPACE_
## 0.1.0a0 (2024-11-25) ## 0.1.0a0 (2024-11-25)
### Feat ### Feat

View file

@ -1,48 +0,0 @@
# SOME DESCRIPTIVE TITLE.
# Copyright (C) YEAR ORGANIZATION
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
#
msgid ""
msgstr ""
"Project-Id-Version: \n"
"POT-Creation-Date: 2025-10-04 17:04+0200\n"
"PO-Revision-Date: 2025-10-04 17:04+0200\n"
"Last-Translator: \n"
"Language-Team: \n"
"Language: fr\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Generated-By: pygettext.py 1.5\n"
"X-Generator: Poedit 3.7\n"
#: src/rougail/user_data_environment/annotator.py:46
msgid ""
"family name must be a lowercase name when we want to use user data "
"\"environment\", so \"{0}\" is invalid"
msgstr ""
"le nom d'une famille doit être un nom en minuscule lorsque vous voulez "
"utiliser de données utilisateur \"environment\", donc \"{0}\" est invalide"
#: src/rougail/user_data_environment/annotator.py:56
msgid ""
"variable name must be a lowercase name when we want to use user data "
"\"environment\", so \"{0}\" is invalid"
msgstr ""
"le nom d'une variable doit être un nom en minuscule lorsque vous voulez "
"utiliser de données utilisateur \"environment\", donc \"{0}\" est invalide"
#: src/rougail/user_data_environment/config.py:34
msgid "Configuration loading environment variables"
msgstr "Configuration le chargement des variables d'environnements"
#: src/rougail/user_data_environment/config.py:42
msgid "Name of the default environment prefix"
msgstr "Nom du préfix d'environnement par défaut"
#: src/rougail/user_data_environment/config.py:49
msgid "Environnement variables may contain secrets"
msgstr "Les variables d'environnement peuvent contenir des secrets"
#~ msgid "Define values from the environment"
#~ msgstr "Défini les valeurs pour l'environnemnt"

View file

@ -1,37 +0,0 @@
# SOME DESCRIPTIVE TITLE.
# Copyright (C) YEAR ORGANIZATION
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
#
msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2025-10-04 17:05+0200\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Generated-By: pygettext.py 1.5\n"
#: src/rougail/user_data_environment/annotator.py:46
msgid "family name must be a lowercase name when we want to use user data \"environment\", so \"{0}\" is invalid"
msgstr ""
#: src/rougail/user_data_environment/annotator.py:56
msgid "variable name must be a lowercase name when we want to use user data \"environment\", so \"{0}\" is invalid"
msgstr ""
#: src/rougail/user_data_environment/config.py:34
msgid "Configuration loading environment variables"
msgstr ""
#: src/rougail/user_data_environment/config.py:42
msgid "Name of the default environment prefix"
msgstr ""
#: src/rougail/user_data_environment/config.py:49
msgid "Environnement variables may contain secrets"
msgstr ""

View file

@ -4,7 +4,7 @@ requires = ["flit_core >=3.8.0,<4"]
[project] [project]
name = "rougail.user_data_environment" name = "rougail.user_data_environment"
version = "0.1.0a15" version = "0.1.0a0"
authors = [{name = "Emmanuel Garette", email = "gnunux@gnunux.info"}] authors = [{name = "Emmanuel Garette", email = "gnunux@gnunux.info"}]
readme = "README.md" readme = "README.md"
description = "Rougail user_data environment" description = "Rougail user_data environment"
@ -13,10 +13,11 @@ license = {file = "LICENSE"}
classifiers = [ classifiers = [
"License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)", "License :: OSI Approved :: GNU Library or Lesser General Public License (LGPL)",
"Programming Language :: Python", "Programming Language :: Python",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
"Programming Language :: Python :: 3.10",
"Programming Language :: Python :: 3.11", "Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12", "Programming Language :: Python :: 3.12",
"Programming Language :: Python :: 3.13",
"Programming Language :: Python :: 3.14",
"Programming Language :: Python :: 3", "Programming Language :: Python :: 3",
"Operating System :: OS Independent", "Operating System :: OS Independent",
"Natural Language :: English", "Natural Language :: English",
@ -35,9 +36,5 @@ name = "cz_conventional_commits"
tag_format = "$version" tag_format = "$version"
version_scheme = "pep440" version_scheme = "pep440"
version_provider = "pep621" version_provider = "pep621"
version_files = [
"src/rougail/user_data_environment/__version__.py",
"pyproject.toml:version"
]
update_changelog_on_bump = true update_changelog_on_bump = true
changelog_merge_prerelease = true changelog_merge_prerelease = true

View file

@ -1,6 +1,3 @@
from .data import RougailUserDataEnvironment from .data import RougailUserDataEnvironment
from .__version__ import __version__
RougailUserData = RougailUserDataEnvironment RougailUserData = RougailUserDataEnvironment
__all__ = ("RougailUserDataEnvironment",) __all__ = ('RougailUserDataEnvironment',)

View file

@ -1 +0,0 @@
__version__ = "0.1.0a15"

View file

@ -1,6 +1,6 @@
""" """
Silique (https://www.silique.fr) Silique (https://www.silique.fr)
Copyright (C) 2024-2025 Copyright (C) 2024
distribued with GPL-2 or later license distribued with GPL-2 or later license
@ -18,7 +18,6 @@ You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
""" """
from rougail.error import DictConsistencyError from rougail.error import DictConsistencyError
from rougail.annotator.variable import Walk from rougail.annotator.variable import Walk
from .i18n import _ from .i18n import _
@ -43,19 +42,11 @@ class Annotator(Walk):
def check_family(self): def check_family(self):
for family in self.get_families(): for family in self.get_families():
if family.name != family.name.lower(): if family.name != family.name.lower():
msg = _( msg = _('family name must be a lowercase name when we want to use user data "environment", so "{0}" is invalid')
'family name must be a lowercase name when we want to use user data "environment", so "{0}" is invalid' raise DictConsistencyError(msg.format(family.name), 200, family.xmlfiles)
)
raise DictConsistencyError(
msg.format(family.name), 200, family.xmlfiles
)
def check_variable(self): def check_variable(self):
for variable in self.get_variables(): for variable in self.get_variables():
if variable.name != variable.name.lower(): if variable.name != variable.name.lower():
msg = _( msg = _('variable name must be a lowercase name when we want to use user data "environment", so "{0}" is invalid')
'variable name must be a lowercase name when we want to use user data "environment", so "{0}" is invalid' raise DictConsistencyError(msg.format(variable.name), 201, variable.xmlfiles)
)
raise DictConsistencyError(
msg.format(variable.name), 201, variable.xmlfiles
)

View file

@ -2,7 +2,7 @@
Config file for Rougail-user-data-environment Config file for Rougail-user-data-environment
Silique (https://www.silique.fr) Silique (https://www.silique.fr)
Copyright (C) 2024-2025 Copyright (C) 2024
distribued with GPL-2 or later license distribued with GPL-2 or later license
@ -22,49 +22,31 @@ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
""" """
from .i18n import _ def get_rougail_config(*,
def get_rougail_config(
*,
backward_compatibility=True, backward_compatibility=True,
) -> dict: ) -> dict:
options = f""" options = """
environment: environment:
description: {_("Configuration loading environment variables")} description: Define values from the environment
disabled: disabled:
jinja: | jinja: |
{{% if _.step.user_data is propertyerror or 'environment' not in _.step.user_data %}} {% if 'environment' not in step.user_data %}
disabled disabled
{{% endif %}} {% endif %}
default_environment_name: default_environment_name:
description: {_("Name of the default environment prefix")} description: Name of the default environment prefix
default: ROUGAIL default: rougail
validators:
- jinja: |-
{{% if _.default_environment_name | upper != _.default_environment_name %}}
{_("should only user uppercase character")}
{{% endif %}}
disabled: disabled:
variable: __.main_namespace jinja: |
when_not: null {% if main_namespace %}
use namespaces
custom_separator: {% endif %}
description: {_('Replace the separator character "." in path by an other')}
help: {_('The "." character could be not allowed in path name')}
mandatory: false
with_secrets:
description: {_("Environnement variables may contain secrets")}
default: true
""" """
return { return {'name': 'environment',
"name": "environment", 'process': 'user data',
"process": "user data", 'options': options,
"options": options, 'level': 50,
"level": 50,
} }
__all__ = ("get_rougail_config",) __all__ = ('get_rougail_config',)

View file

@ -1,6 +1,6 @@
""" """
Silique (https://www.silique.fr) Silique (https://www.silique.fr)
Copyright (C) 2024-2025 Copyright (C) 2024
distribued with GPL-2 or later license distribued with GPL-2 or later license
@ -18,18 +18,14 @@ You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
""" """
import os import os
from rougail.tiramisu import CONVERT_OPTION from rougail.object_model import CONVERT_OPTION
from rougail.config import RougailConfig from rougail.config import RougailConfig
from rougail.error import ExtensionError
from tiramisu.error import ValueOptionError from tiramisu.error import ValueOptionError
class RougailUserDataEnvironment: class RougailUserDataEnvironment:
def __init__( def __init__(self,
self, config: 'Config',
config: "Config",
*, *,
rougailconfig: RougailConfig=None, rougailconfig: RougailConfig=None,
): ):
@ -37,71 +33,60 @@ class RougailUserDataEnvironment:
self.config = config self.config = config
if rougailconfig is None: if rougailconfig is None:
rougailconfig = RougailConfig rougailconfig = RougailConfig
user_data = rougailconfig["step.user_data"] user_data = rougailconfig['step.user_data']
if "environment" not in user_data: if 'environment' not in user_data:
user_data.append("environment") user_data.append('environment')
rougailconfig["step.user_data"] = user_data rougailconfig['step.user_data'] = user_data
user_data = rougailconfig["step.user_data"] user_data = rougailconfig['step.user_data']
if "environment" not in user_data: self.rougailconfig = rougailconfig
raise ExtensionError("environment is not set in step.user_data") if 'environment' not in user_data:
if "environment.with_secrets" in rougailconfig: raise Exception('environment is not set in step.user_data')
self.with_secrets = rougailconfig["environment.with_secrets"]
else:
self.with_secrets = True
self.custom_separator = rougailconfig["environment.custom_separator"]
if not rougailconfig["main_namespace"]:
self.default_environment_name = rougailconfig["environment.default_environment_name"]
self.errors = [] self.errors = []
self.warnings = [] self.warnings = []
def run(self): def run(self):
values = self.parse() values = self.parse()
return [ return [{'source': 'environment',
{ 'errors': self.errors,
"source": "environment variable", 'warnings': self.warnings,
"errors": self.errors, 'values': values,
"warnings": self.warnings, 'options': {'multi_separator': ',',
"values": values, 'needs_convert': True,
"options": {
"multi_separator": ",",
"needs_convert": True,
"allow_secrets_variables": self.with_secrets,
}, },
} }]
]
def parse(self): def parse(self):
variables = {} variables = {}
self.prefixes = [] found_ns = False
for option in self.config: for option in self.config:
if not option.isoptiondescription() or option.group_type() != "namespace": if option.group_type() == "namespace":
break found_ns = True
self.prefixes.append(option.name().upper() + ".") variables.update(get_rougail_environment(option.name()))
else: if not found_ns:
return self.get_rougail_environment(0) if self.rougailconfig['main_namespace'] is None:
# no namespace then we filter the ROUGAIL_ environment variables return get_rougail_environment(self.rougailconfig['environment.default_environment_name'])
self.prefixes = [self.default_environment_name + "_"] return get_rougail_environment(self.rougailconfig['main_namespace'])
return self.get_rougail_environment(len(self.prefixes[0])) return variables
def get_rougail_environment(self, len_env):
def get_rougail_environment(namespace):
"""gets all the rougail environment variables and their values """gets all the rougail environment variables and their values
:sample: {'VARINT': '5', 'VARNAME34': '58, 22', 'VARNAME2': 'tata', :sample: {'VARINT': '5', 'VARNAME34': '58, 22', 'VARNAME2': 'tata',
'VARNAME1': 'titi', 'MYFAMILY.VARNAME3': 'spam'} 'VARNAME1': 'titi', 'MYFAMILY.VARNAME3': 'spam'}
:returns: rougail environment variables as a key/value dict :returns: rougail environment variables as a key/value dict
""" """
return { # first we look at all environment variables
envvar[len_env:].lower(): envval all_envvar = os.environ
for envvar, envval in self.get_correct_envs() # then we filter the ROUGAIL_ environment variables
} if namespace is None:
rougail_environment_var = rougail_default_environment_var.upper() + '_'
def get_correct_envs(self): len_env = len(rougail_environment_var) + 1
for envvar, envval in os.environ.items(): root = ''
if self.custom_separator: else:
envvar = envvar.replace(self.custom_separator, '.') rougail_environment_var = namespace.upper() + '_'
for prefix in self.prefixes: len_env = len(rougail_environment_var)
if envvar.startswith(prefix): root = namespace.lower() + '.'
yield envvar, envval return {root + envvar[len_env:].lower(): envval
for prefix in self.prefixes: for envvar, envval in all_envvar.items()
if envvar.startswith(prefix): if envvar.startswith(rougail_environment_var)}
yield envvar, envval

View file

@ -1,27 +0,0 @@
"""Internationalisation utilities
Silique (https://www.silique.fr)
Copyright (C) 2025
This program is free software: you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published by the
Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
This program is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
details.
You should have received a copy of the GNU Lesser General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
from gettext import translation
from pathlib import Path
t = translation(
"rougail_user_data_environment", str(Path(__file__).parent / "locale"), fallback=True
)
_ = t.gettext

View file

@ -1 +0,0 @@
ROUGAIL_UNKNOWN=1

View file

@ -1,2 +0,0 @@
ROUGAIL_FAMILY_DISABLED.VARIABLE1=test
ROUGAIL_FAMILY_DISABLED.VARIABLE2=test

View file

@ -1 +0,0 @@
ROUGAIL_VARIABLE_DISABLED=test

View file

@ -1 +0,0 @@
ROUGAIL_VARIABLE_INT=test

View file

@ -1 +0,0 @@
ROUGAIL_FAMILY_DISABLED_HIDDEN.VARIABLE1=test

View file

@ -1 +0,0 @@
ROUGAIL_SECRET=a

View file

@ -1 +0,0 @@
ROUGAIL_FAMILY=test

View file

@ -1 +0,0 @@
ROUGAIL_DYN_UNKNOWN.VARIABLE=test

View file

@ -1,4 +0,0 @@
ROUGAIL_LEADERSHIP.LEADER=test1,test2,test3
ROUGAIL_LEADERSHIP.FOLLOWER=test1,test2,test3
ROUGAIL_LEADERSHIP.FOLLOWER_DISABLED=test1,test2,test3
ROUGAIL_LEADERSHIP.FOLLOWER_DISABLED_AT_INDEX=test1,test2,test3

View file

@ -1,2 +0,0 @@
ROUGAIL_LEADERSHIP.LEADER=test
ROUGAIL_LEADERSHIP.UNKNWON=test

View file

@ -1,2 +0,0 @@
ROUGAIL_LEADERSHIP.LEADER=test
ROUGAIL_LEADERSHIP.FOLLOWER=test1,test2

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"variable or family \"unknown\" does not exist, it will be ignored when loading from environment variable"
]
}

View file

@ -1,7 +0,0 @@
{
"errors": [],
"warnings": [
"family \"family_disabled\" is disabled, \"family_disabled.variable1\" will be ignored when loading from environment variable",
"family \"family_disabled\" is disabled, \"family_disabled.variable2\" will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"variable \"variable_disabled\" is disabled, it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"the value \"test\" is an invalid integer for \"variable_int\", which is not an integer, it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"family \"family_disabled_hidden\" is disabled and hidden, \"family_disabled_hidden.variable1\" will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"the value \"**********\" is an invalid password for \"secret\", at least 10 characters are required, it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"cannot set the value \"test\" to the family \"family\", it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"variable or family \"dyn_unknown\" does not exist, it will be ignored when loading from environment variable"
]
}

View file

@ -1,9 +0,0 @@
{
"errors": [],
"warnings": [
"variable \"leadership.follower_disabled\" at index \"0\" is disabled, it will be ignored when loading from environment variable",
"variable \"leadership.follower_disabled\" at index \"1\" is disabled, it will be ignored when loading from environment variable",
"variable \"leadership.follower_disabled\" at index \"2\" is disabled, it will be ignored when loading from environment variable",
"variable \"leadership.follower_disabled_at_index\" at index \"1\" is disabled, it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"variable or family \"leadership.unknwon\" does not exist, it will be ignored when loading from environment variable"
]
}

View file

@ -1,6 +0,0 @@
{
"errors": [],
"warnings": [
"index \"1\" is greater than the leadership length \"1\" for option \"leadership.follower\" in environment variable"
]
}

View file

@ -1,68 +0,0 @@
%YAML 1.2
---
version: 1.1
variable:
secret:
type: secret
params:
min_len: 10
variable_disabled:
disabled: true
variable_disabled_hidden:
disabled: true
hidden: true
variable_int:
type: number
family:
variable:
family_disabled:
disabled: true
variable1:
variable2:
family_disabled_hidden:
disabled: true
hidden: true
variable1:
variable2:
dyn_{{ identifier }}:
dynamic:
- var1
- var2
variable:
leadership:
type: leadership
leader:
follower:
follower_disabled:
disabled: true
follower_disabled_at_index:
disabled:
jinja: >-
{% if index == 1 %}
true
{% else %}
{% endif %}
params:
index:
type: index
...

View file

@ -0,0 +1,3 @@
{
"rougail.var1": "string1"
}

View file

@ -0,0 +1,3 @@
{
"rougail.var1": "string1"
}

Some files were not shown because too many files have changed in this diff Show more