Metadata-Version: 2.1
Name: mcddns
Version: 1.2.0
Summary: Industrial-strength dynamic DNS client
Home-page: https://github.com/mconigliaro/mcddns
Author: Mike Conigliaro
Author-email: mike@conigliaro.org
Requires-Python: >=3.8,<4.0
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Console
Classifier: Intended Audience :: Information Technology
Classifier: Intended Audience :: System Administrators
Classifier: License :: OSI Approved :: MIT License
Classifier: Natural Language :: English
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.8
Classifier: Programming Language :: Python :: 3.9
Classifier: Topic :: Internet :: Name Service (DNS)
Classifier: Topic :: System :: Systems Administration
Requires-Dist: boto3 (>=1.23,<2.0)
Requires-Dist: requests (>=2.27,<3.0)
Project-URL: Repository, https://github.com/mconigliaro/mcddns
Description-Content-Type: text/markdown

# MCDDNS

[![Continuous Integration](https://github.com/mconigliaro/mcddns/actions/workflows/ci.yml/badge.svg)](https://github.com/mconigliaro/mcddns/actions/workflows/ci.yml)

**M**ike **C**onigliaro's industrial-strength **D**ynamic **D**omain **N**ame **S**ystem client

## Features

- Extensible plugin-oriented architecture with a simple API
  - Address providers are responsible for obtaining an address
  - DNS providers are responsible for managing a DNS record
- Resilient against network and endpoint failures
  - Built-in result validation for address providers
  - Use multiple address providers (keep trying until one succeeds)
  - Optional retry with Fibonacci backoff
  - Cron mode (prevents email flood if your Internet connections goes down)
- Detailed logging
- Dry-run mode (shows what will happen without changing anything)

### Built-In Providers

#### Address Providers

- `aws.CheckIP`: Obtains a public IPv4 address via [Amazon Web Services](https://aws.amazon.com/)
- `dyn.CheckIP`: Obtains a public IPv4 address via [Dyn](https://dyn.com/)
- `google.CheckIP`: Obtains a public IPv4 address via [Google Domains](https://domains.google.com)
- `ipify.IPv4`: Obtains a public IPv4 address via [ipify](https://www.ipify.org/)
- `ipify.IPv6`: Obtains an IPv6 address via [ipify](https://www.ipify.org/)

#### DNS Providers

- `aws.Route53`: Manages records in [Amazon Route53](https://aws.amazon.com/route53/)

## Installation

    pip install mcddns

## Running the Application

    mcddns <dns_provider> <fqdn> [options]

Use `--help` to see available options.

## Development

### Getting Started

    poetry install
    poetry shell
    ...

### Running Tests

    pytest

### Writing Providers

A provider is any class that inherits from `AddressProvider` or `DNSProvider`. In production mode, Python modules/packages prefixed with `mcddns_` will automatically be imported from the following locations:

1. `$XDG_CONFIG_HOME/mcddns/provider`
1. `/etc/mcddns/provider`
1. [sys.path](https://docs.python.org/3/library/sys.html#sys.path)

Example: If you create a file at `$XDG_CONFIG_HOME/mcddns/provider/mcddns_foo.py` with a class named `Bar` that inherits from one of the `Provider` subclasses, your module can be referenced (e.g. in command-line options) as `foo.Bar`.

#### Provider Methods

Each provider type has a set of methods that will be called in a particular order. Note that some of these methods are expected to return a specific value in order to progress to the next step.

##### All Providers

1. `options_pre(parser)`: Runs before option parsing. Use this method to add your own provider-specific command line arguments (See: [argparse](https://docs.python.org/3.6/library/argparse.html)).
1. `options_post(parser, options)`: Runs after option parsing. Use this method to do things with your provider-specific command line arguments.

##### Address Providers

1. `fetch(options)`: Fetches and returns an IP address, hostname, etc.
1. `validate(options, address)`: Returns `True` if the address is valid and `False` otherwise

##### DNS Providers

1. `check(options, address)`: Returns `True` if a DNS update is required and `False` otherwise
1. `update(options, address)`: Returns `True` if a DNS update was successful and `False` otherwise

#### Examples

I'll write more documentation if people are interested, but for now, see the examples at [mcddns/provider](mcddns/provider) and [tests/provider](tests/provider).

#### Releases

1. Bump `version` in [pyproject.toml](pyproject.toml)
1. Update [CHANGELOG.md](CHANGELOG.md)
1. Run `make release`

### To Do

- Add tests for built-in providers

