Config-OC


NameConfig-OC JSON
Version 1.0.1 PyPI version JSON
download
home_pagehttps://ouroboroscoding.com/config/
SummaryHandles loading loading configuration files based on hostname
upload_time2023-06-27 11:15:28
maintainer
docs_urlNone
authorChris Nasr - Ouroboros Coding Inc.
requires_python>=3.10
licenseMIT
keywords config configuration host specific configuration
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # Ouroboros Coding Config
[![pypi version](https://img.shields.io/pypi/v/Config-OC.svg)](https://pypi.org/project/Config-OC) ![MIT License](https://img.shields.io/pypi/l/Config-OC.svg)

A module to simplify documenting and loading configuration files written in JSON with the ability to seperate public from private data.

# Requires
Config-OC requires python 3.10 or higher

## Installation
```bash
pip install Config-OC
```

# Usage

## Creating Files

Using the module does not actually require any files, however it is strongly suggested both are created to get the full benefits of the module. Not doing so will result in an error printed to stderr, but the module will still work as intended, just with no settings.

### Base Settings

The first file is always called config.json. Its purpose is to provide a document detailing all possible settings, as valid types, if not necessarily functional values.

config.json:
```json
{
  "mariadb": {
    "host": "localhost",
    "port": 3306,
    "user": "foobar_user",
    "passwd": "",
    "db": "foobar",
    "charset": "utf8"
  },

  "redis": {
    "host": "localhost",
    "port": 6379,
    "password": "",
    "db": 0
  }
}
```

Notice, we don't put any password in here, these value do not need to work, they just need to provide information to the developer. In this case, that the config.mariadb.password value needs to be a string. This is a safe way to provide documentation, as well as safe default values, that can be stored in source control and shared publicly.

### Host Specific Settings

The second file is named based on the hostname of the machine the script is running on. It allows you to set host specific, or private, settings in a file associated only with the host. It should never be stored in source control, and thus never shared publicly, even accidently.

It also starts with config, and ends in .json but contains the name of the host in the middle. For example, say my development machine was called "chris"

```bash
$: echo `hostname`
chris
```

I would then create my specific settings file as config.chris.json. An easy way to do so from the command line would be using the command used above to generate the filename

```bash
$: touch config.`hostname`.json
$: ls
config.chris.json
```

This file contains only the changes you want to see on top of the base settings. For example, I am still on my development machine, and MariaDB is running locally, on the default port, with the default databases name, and the default database user, with the default charset. Sounds like all I need is to set the password. In fact, same deal for Redis, my default settings are good for all but password. So let's set our config.chris.json

config.chris.json
```json
{
  "mariadb": {
    "passwd": "d7e8fuisdjf02233"
  },

  "redis": {
    "password": "aepof20rif323t23"
  }
}
```

But what if I was not on my development machine, what if I was on a staging server, called "preprod" and both softwares are still local, still on the default port, but using a different db, user, and password?

config.preprod.json
```json
{
  "mariadb": {
    "user": "staging_user",
    "passwd": "g38s5h2k1ng38dby",
    "db": "staging"
  },

  "redis": {
    "password": "4ng8sl26flv3s8hs",
    "db": 1
  }
}
```

And, last example, we're in production, on our thundercougarfalconbird server, and the hosts of both softwares is external, and the passwords are needed, but the rest is fine as is.

config.thundercougarfalconbird.json
```json
{
  "mariadb": {
    "host": "db.somedomain.com",
    "passwd": "4kf8an38d8nf4alf0"
  },

  "redis": {
    "host": "cache.somedomain.com",
    "password": "f8askgwk9shostfd"
  }
}
```

## Using config
So now we have our configuration, but we need access to the data. First, let's
see how we can get all of it.

script.py:
```python
from config import config
from pprint import pprint

pprint(
  config()
)
```

If we we on my dev machine, the output would be something like this

```bash
{'mariadb': {'charset': 'utf8',
             'db': 'foobar',
             'host': 'localhost',
             'passwd': 'd7e8fuisdjf02233',
             'port': 3306,
             'user': 'foobar_user'},
 'redis': {'db': 0,
           'host': 'localhost',
           'password': 'aepof20rif323t23',
           'port': 6379}}
```

If we were on the production server, the output would be something like this

```bash
{'mariadb': {'charset': 'utf8',
             'db': 'foobar',
             'host': 'db.somedomain.com',
             'passwd': '4kf8an38d8nf4alf0',
             'port': 3306,
             'user': 'foobar_user'},
 'redis': {'db': 0,
           'host': 'cache.somedomain.com',
           'password': 'f8askgwk9shostfd',
           'port': 6379}
}
```

But it's not at all necessary to get the entire dictionary of settings, nor is config even a function really. It's simply a shorthand to get a copy of all settings. Nothing stops you from getting just one section (dev)

script.py:
```python
pprint(
  config.mariadb()
)
```

```bash
{'db': 'foobar',
 'host': 'localhost',
 'password': 'd7e8fuisdjf02233',
 'port': 3306,
 'user': 'foobar_user'
}
```

Or even just one value

script.py:
```python
pprint(
  config.mariadb.host()
)
```

```bash
'localhost'
```

So why the function call? Why not just return the data at the level requested and be done with it? Well, because then we couldn't have another level of security by allowing the developer to specify exactly what is necessary, regardless of the settings file. I mean, what if someone forgets a value? Should your code crash because something is missing in a text file? Or maybe, something like a port, is so generic, and so likely to be the default, that you want to allow the opportunity to change it, but you're not advertising it because you want to push people to just stick with the default port? In fact, for Redis, there's a good chance the db and the port are never going to change, let's not bog down the user with these options.

config.json:
```json
{
  "redis": {
    "host": "localhost",
    "password": ""
  }
}
```

config.chris.json:
```json
{
  "redis": {
    "password": "aepof20rif323t23"
  }
}
```

script.py:
```python
from config import config
from pprint import pprint

pprint(
  config.redis({
    'host': 'localhost',
    'port': 6379,
    'password': '',
    'db': 0
  })
)
```

```bash
{'db': 0,
 'host': 'localhost',
 'password': 'aepof20rif323t23',
 'port': 6379
}
```

And there's no reason why you need to access the data immediately, or as an entire section. What if I just want to pull out a part of the data and pass it along before I make any decisions about what defaults it needs, or even what parts I need?

script.py:
```python
from config import config, Data

def print_settings(conf: 'Data') -> None:
  print('All: %s' % str(conf()))
  print('Host: %s' % conf.host('localhost'))
  print('Port: %d' % conf.port(6379))
  print('DB: %d' % conf.db(0))

redis_conf = config.redis
print_settings(redis_conf)
```

```bash
All: {'db': 0, 'host': 'localhost', 'password': 'aepof20rif323t23', 'port': 6379}
Host: localhost
Port: 6379
DB: 0
```

And this works regardless of what you've tried to access or whether it exists or not. As an example, there is no section for logging in our original config, or the host specific ones. Logging is never even mentioned.

script.py:
```python
from config import config, Data

def print_settings(conf: 'Data') -> None:
  print('All: %s' % str(conf()))
  print('Name template: %s' % conf.name_template('file.%d.log'))
  print('Max size: %s' % conf.max_size(10485760))
  print('Max files: %d' % conf.max_files(10))

logging_conf = config.logging
print_settings(logging_conf)
```

```bash
All: None
Name template: file.%d.log
Max size: 10485760
Max files: 10
```

So why did this code work? There's no "logging" in our configuration files, and we didn't add it at runtime, shouldn't this throw a exception? Maybe an AttributeError? Sure, it could have been designed that way, but then you'd always be stuck in a place where you need to decide whether you want to make something configurable or not. Maybe you just want the option, in the future, but again, you're not advertising it to the world. Maybe it's a beta feature, but should you personally have to commit, push, update, test, commit, push, updated, test, again and again to adjust something?


            

Raw data

            {
    "_id": null,
    "home_page": "https://ouroboroscoding.com/config/",
    "name": "Config-OC",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": "",
    "keywords": "config,configuration,host specific configuration",
    "author": "Chris Nasr - Ouroboros Coding Inc.",
    "author_email": "chris@ouroboroscoding.com",
    "download_url": "https://files.pythonhosted.org/packages/89/85/3b3a9f6c58e862be4f9f08120dddfd27fd8cc96fd13bc460b06ebee6ddb2/Config-OC-1.0.1.tar.gz",
    "platform": null,
    "description": "# Ouroboros Coding Config\n[![pypi version](https://img.shields.io/pypi/v/Config-OC.svg)](https://pypi.org/project/Config-OC) ![MIT License](https://img.shields.io/pypi/l/Config-OC.svg)\n\nA module to simplify documenting and loading configuration files written in JSON with the ability to seperate public from private data.\n\n# Requires\nConfig-OC requires python 3.10 or higher\n\n## Installation\n```bash\npip install Config-OC\n```\n\n# Usage\n\n## Creating Files\n\nUsing the module does not actually require any files, however it is strongly suggested both are created to get the full benefits of the module. Not doing so will result in an error printed to stderr, but the module will still work as intended, just with no settings.\n\n### Base Settings\n\nThe first file is always called config.json. Its purpose is to provide a document detailing all possible settings, as valid types, if not necessarily functional values.\n\nconfig.json:\n```json\n{\n  \"mariadb\": {\n    \"host\": \"localhost\",\n    \"port\": 3306,\n    \"user\": \"foobar_user\",\n    \"passwd\": \"\",\n    \"db\": \"foobar\",\n    \"charset\": \"utf8\"\n  },\n\n  \"redis\": {\n    \"host\": \"localhost\",\n    \"port\": 6379,\n    \"password\": \"\",\n    \"db\": 0\n  }\n}\n```\n\nNotice, we don't put any password in here, these value do not need to work, they just need to provide information to the developer. In this case, that the config.mariadb.password value needs to be a string. This is a safe way to provide documentation, as well as safe default values, that can be stored in source control and shared publicly.\n\n### Host Specific Settings\n\nThe second file is named based on the hostname of the machine the script is running on. It allows you to set host specific, or private, settings in a file associated only with the host. It should never be stored in source control, and thus never shared publicly, even accidently.\n\nIt also starts with config, and ends in .json but contains the name of the host in the middle. For example, say my development machine was called \"chris\"\n\n```bash\n$: echo `hostname`\nchris\n```\n\nI would then create my specific settings file as config.chris.json. An easy way to do so from the command line would be using the command used above to generate the filename\n\n```bash\n$: touch config.`hostname`.json\n$: ls\nconfig.chris.json\n```\n\nThis file contains only the changes you want to see on top of the base settings. For example, I am still on my development machine, and MariaDB is running locally, on the default port, with the default databases name, and the default database user, with the default charset. Sounds like all I need is to set the password. In fact, same deal for Redis, my default settings are good for all but password. So let's set our config.chris.json\n\nconfig.chris.json\n```json\n{\n  \"mariadb\": {\n    \"passwd\": \"d7e8fuisdjf02233\"\n  },\n\n  \"redis\": {\n    \"password\": \"aepof20rif323t23\"\n  }\n}\n```\n\nBut what if I was not on my development machine, what if I was on a staging server, called \"preprod\" and both softwares are still local, still on the default port, but using a different db, user, and password?\n\nconfig.preprod.json\n```json\n{\n  \"mariadb\": {\n    \"user\": \"staging_user\",\n    \"passwd\": \"g38s5h2k1ng38dby\",\n    \"db\": \"staging\"\n  },\n\n  \"redis\": {\n    \"password\": \"4ng8sl26flv3s8hs\",\n    \"db\": 1\n  }\n}\n```\n\nAnd, last example, we're in production, on our thundercougarfalconbird server, and the hosts of both softwares is external, and the passwords are needed, but the rest is fine as is.\n\nconfig.thundercougarfalconbird.json\n```json\n{\n  \"mariadb\": {\n    \"host\": \"db.somedomain.com\",\n    \"passwd\": \"4kf8an38d8nf4alf0\"\n  },\n\n  \"redis\": {\n    \"host\": \"cache.somedomain.com\",\n    \"password\": \"f8askgwk9shostfd\"\n  }\n}\n```\n\n## Using config\nSo now we have our configuration, but we need access to the data. First, let's\nsee how we can get all of it.\n\nscript.py:\n```python\nfrom config import config\nfrom pprint import pprint\n\npprint(\n  config()\n)\n```\n\nIf we we on my dev machine, the output would be something like this\n\n```bash\n{'mariadb': {'charset': 'utf8',\n             'db': 'foobar',\n             'host': 'localhost',\n             'passwd': 'd7e8fuisdjf02233',\n             'port': 3306,\n             'user': 'foobar_user'},\n 'redis': {'db': 0,\n           'host': 'localhost',\n           'password': 'aepof20rif323t23',\n           'port': 6379}}\n```\n\nIf we were on the production server, the output would be something like this\n\n```bash\n{'mariadb': {'charset': 'utf8',\n             'db': 'foobar',\n             'host': 'db.somedomain.com',\n             'passwd': '4kf8an38d8nf4alf0',\n             'port': 3306,\n             'user': 'foobar_user'},\n 'redis': {'db': 0,\n           'host': 'cache.somedomain.com',\n           'password': 'f8askgwk9shostfd',\n           'port': 6379}\n}\n```\n\nBut it's not at all necessary to get the entire dictionary of settings, nor is config even a function really. It's simply a shorthand to get a copy of all settings. Nothing stops you from getting just one section (dev)\n\nscript.py:\n```python\npprint(\n  config.mariadb()\n)\n```\n\n```bash\n{'db': 'foobar',\n 'host': 'localhost',\n 'password': 'd7e8fuisdjf02233',\n 'port': 3306,\n 'user': 'foobar_user'\n}\n```\n\nOr even just one value\n\nscript.py:\n```python\npprint(\n  config.mariadb.host()\n)\n```\n\n```bash\n'localhost'\n```\n\nSo why the function call? Why not just return the data at the level requested and be done with it? Well, because then we couldn't have another level of security by allowing the developer to specify exactly what is necessary, regardless of the settings file. I mean, what if someone forgets a value? Should your code crash because something is missing in a text file? Or maybe, something like a port, is so generic, and so likely to be the default, that you want to allow the opportunity to change it, but you're not advertising it because you want to push people to just stick with the default port? In fact, for Redis, there's a good chance the db and the port are never going to change, let's not bog down the user with these options.\n\nconfig.json:\n```json\n{\n  \"redis\": {\n    \"host\": \"localhost\",\n    \"password\": \"\"\n  }\n}\n```\n\nconfig.chris.json:\n```json\n{\n  \"redis\": {\n    \"password\": \"aepof20rif323t23\"\n  }\n}\n```\n\nscript.py:\n```python\nfrom config import config\nfrom pprint import pprint\n\npprint(\n  config.redis({\n    'host': 'localhost',\n    'port': 6379,\n    'password': '',\n    'db': 0\n  })\n)\n```\n\n```bash\n{'db': 0,\n 'host': 'localhost',\n 'password': 'aepof20rif323t23',\n 'port': 6379\n}\n```\n\nAnd there's no reason why you need to access the data immediately, or as an entire section. What if I just want to pull out a part of the data and pass it along before I make any decisions about what defaults it needs, or even what parts I need?\n\nscript.py:\n```python\nfrom config import config, Data\n\ndef print_settings(conf: 'Data') -> None:\n  print('All: %s' % str(conf()))\n  print('Host: %s' % conf.host('localhost'))\n  print('Port: %d' % conf.port(6379))\n  print('DB: %d' % conf.db(0))\n\nredis_conf = config.redis\nprint_settings(redis_conf)\n```\n\n```bash\nAll: {'db': 0, 'host': 'localhost', 'password': 'aepof20rif323t23', 'port': 6379}\nHost: localhost\nPort: 6379\nDB: 0\n```\n\nAnd this works regardless of what you've tried to access or whether it exists or not. As an example, there is no section for logging in our original config, or the host specific ones. Logging is never even mentioned.\n\nscript.py:\n```python\nfrom config import config, Data\n\ndef print_settings(conf: 'Data') -> None:\n  print('All: %s' % str(conf()))\n  print('Name template: %s' % conf.name_template('file.%d.log'))\n  print('Max size: %s' % conf.max_size(10485760))\n  print('Max files: %d' % conf.max_files(10))\n\nlogging_conf = config.logging\nprint_settings(logging_conf)\n```\n\n```bash\nAll: None\nName template: file.%d.log\nMax size: 10485760\nMax files: 10\n```\n\nSo why did this code work? There's no \"logging\" in our configuration files, and we didn't add it at runtime, shouldn't this throw a exception? Maybe an AttributeError? Sure, it could have been designed that way, but then you'd always be stuck in a place where you need to decide whether you want to make something configurable or not. Maybe you just want the option, in the future, but again, you're not advertising it to the world. Maybe it's a beta feature, but should you personally have to commit, push, update, test, commit, push, updated, test, again and again to adjust something?\n\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "Handles loading loading configuration files based on hostname",
    "version": "1.0.1",
    "project_urls": {
        "Documentation": "https://ouroboroscoding.com/config/",
        "Homepage": "https://ouroboroscoding.com/config/",
        "Source": "https://github.com/ouroboroscoding/config-python",
        "Tracker": "https://github.com/ouroboroscoding/config-python/issues"
    },
    "split_keywords": [
        "config",
        "configuration",
        "host specific configuration"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "89853b3a9f6c58e862be4f9f08120dddfd27fd8cc96fd13bc460b06ebee6ddb2",
                "md5": "a77d9a06fb05a72ffafb3153d36e5fc3",
                "sha256": "abb9888bf9a64488c64e818f3902994ebef3ab2c3dbb590dcd0b34e3bba4344e"
            },
            "downloads": -1,
            "filename": "Config-OC-1.0.1.tar.gz",
            "has_sig": false,
            "md5_digest": "a77d9a06fb05a72ffafb3153d36e5fc3",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 6819,
            "upload_time": "2023-06-27T11:15:28",
            "upload_time_iso_8601": "2023-06-27T11:15:28.197887Z",
            "url": "https://files.pythonhosted.org/packages/89/85/3b3a9f6c58e862be4f9f08120dddfd27fd8cc96fd13bc460b06ebee6ddb2/Config-OC-1.0.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-06-27 11:15:28",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "ouroboroscoding",
    "github_project": "config-python",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "config-oc"
}
        
Elapsed time: 0.08393s