How to Parse INI Config Files in Python with Configparser

Configuration files provide a structured way to manage application settings that’s more organized than environment variables alone.

INI files, short for initialization files, with their simple section-based format, are both easy to read and parse. Python’s built-in configparser module makes working with these files straightforward and powerful.

This tutorial will teach you how to read and parse such .ini config files using the configparser module.

🔗 Here’s the code on GitHub.

Prerequisites

To follow along with this tutorial, you should have:

  • Python 3.7 or later installed on your system

  • Basic understanding of Python syntax and data structures (dictionaries, strings)

  • Familiarity with file operations in Python

  • A text editor or IDE for writing Python code

  • Basic knowledge of configuration files and why they’re used in applications

No external packages are required, as we’ll be using Python’s built-in configparser module.

Table of Contents

  1. Understanding the INI File Format

  2. Basic ConfigParser Usage

  3. Type Conversion and Default Values

  4. How to Create a Simple Config Manager

  5. How to Work with Multiple Sections in INI Files

  6. How to Write Configuration Files

Understanding the INI File Format

INI files organize configuration into sections, where each section contains key-value pairs. This structure is useful for applications with multiple components or environments. Let’s look at what an INI file looks like before we parse it.

Create a file named app.ini:

[database]
host = localhost
port = 5432
username = app_user
password = secure_password
pool_size = 10
ssl_enabled = true

[server]
host = 0.0.0.0
port = 8000
debug = false

[logging]
level = INFO
file = app.log

This file contains three sections: database, server, and logging. Each section groups related settings together, making the configuration easy to understand and maintain.

Basic ConfigParser Usage

The configparser module provides the ConfigParser class, which handles all the parsing work. Here’s how to read and access configuration values:

import configparser

config = configparser.ConfigParser()
config.read('app.ini')

# Access values from sections
db_host = config['database']['host']
db_port = config['database']['port']

print(f"Database: {db_host}:{db_port}")
print(f"Sections: {config.sections()}")

This code shows the basic workflow:

  • create a ConfigParser object,

  • read your INI file,

  • then access values using dictionary-like syntax.

The first bracket contains the section name, and the second contains the key.

Create the app.ini file and run the above code. You should see the following output:

Database: localhost:5432
Sections: ['database', 'server', 'logging']

Type Conversion and Default Values

Configuration values in INI files are stored as strings, but you often need them as integers, booleans, or floats. ConfigParser provides convenient methods for type conversion as shown here:

import configparser

config = configparser.ConfigParser()
config.read('app.ini')

# Automatic type conversion
db_port = config.getint('database', 'port')
ssl_enabled = config.getboolean('database', 'ssl_enabled')

# With fallback defaults
max_retries = config.getint('database', 'max_retries', fallback=3)
timeout = config.getfloat('database', 'timeout', fallback=30.0)

print(f"Port: {db_port}, SSL: {ssl_enabled}")

In this code, the getint(), getboolean(), and getfloat() methods convert string values to the appropriate type. The fallback parameter provides a default value when the key doesn’t exist, preventing errors.

When you run the above code, you’ll get:

Port: 5432, SSL: True

How to Create a Simple Config Manager

A practical approach is to wrap ConfigParser in a class that validates configuration and provides easy access to settings:

import configparser
from pathlib import Path

class ConfigManager:
    def __init__(self, config_file='app.ini'):
        self.config = configparser.ConfigParser()

        if not Path(config_file).exists():
            raise FileNotFoundError(f"Config file not found: {config_file}")

        self.config.read(config_file)

    def get_database_config(self):
        db = self.config['database']
        return {
            'host': db.get('host'),
            'port': db.getint('port'),
            'username': db.get('username'),
            'password': db.get('password'),
            'pool_size': db.getint('pool_size', fallback=5)
        }

This manager class validates that the file exists and provides clean methods to access configuration. It returns dictionaries with properly typed values.

And you can use it like so:

config = ConfigManager('app.ini')
db_config = config.get_database_config()
print(db_config)

This outputs:

{'host': 'localhost', 'port': 5432, 'username': 'app_user', 'password': 'secure_password', 'pool_size': 10}

How to Work with Multiple Sections in INI Files

You can organize different parts of your application into separate sections and access them independently:

import configparser

config = configparser.ConfigParser()
config.read('app.ini')

# Get all options in a section as a dictionary
db_settings = dict(config['database'])
server_settings = dict(config['server'])

# Check if a section exists
if config.has_section('cache'):
    cache_enabled = config.getboolean('cache', 'enabled')
else:
    cache_enabled = False

print(f"Database settings: {db_settings}")
print(f"Caching enabled: {cache_enabled}")

The dict() conversion gives you all key-value pairs from a section at once. The has_section() method lets you conditionally handle optional configuration sections.

Running the above code should give you the following output:

Database settings: {'host': 'localhost', 'port': '5432', 'username': 'app_user', 'password': 'secure_password', 'pool_size': '10', 'ssl_enabled': 'true'}
Caching enabled: False

How to Write Configuration Files

ConfigParser can also create and modify INI files, which is useful for saving user preferences or generating config templates:

import configparser

config = configparser.ConfigParser()

# Add sections and values
config['database'] = {
    'host': 'localhost',
    'port': '5432',
    'username': 'myapp'
}

config['server'] = {
    'host': '0.0.0.0',
    'port': '8000',
    'debug': 'false'
}

# Write to file
with open('generated.ini', 'w') as configfile:
    config.write(configfile)

print("Configuration file created!")

This code creates a new INI file from scratch. The write() method saves the configuration in the proper INI format with sections and key-value pairs.

Conclusion

When environment variables aren’t enough and you need grouped settings for different components, INI files are your answer.

The format is human-readable, ConfigParser handles type conversion automatically, and it’s built into Python’s standard library. Wrap it in a configuration class for validation and clean access patterns.

Also remember:

  • Organize by component. Use sections to group related settings.

  • Use type conversion methods. Always use getint(), getboolean(), and getfloat() rather than manual conversion. They handle edge cases better.

  • Provide sensible defaults. Use the fallback parameter for optional settings so your application works with minimal configuration.

  • Validate early. Check that required sections and keys exist at startup before attempting to use them.

  • Keep secrets separate. Don’t commit INI files with passwords to version control. Use .ini.example files with dummy values as templates.

In the next article, you’ll learn how to work with TOML files in Python. Until then, keep coding!

Powered by WPeMatico