Metadata-Version: 2.4
Name: textswap
Version: 0.2.0
Summary: Bulk text replacement in files using dictionary mappings
Project-URL: Homepage, https://github.com/cainky/textswap
Project-URL: Repository, https://github.com/cainky/textswap
Project-URL: Issues, https://github.com/cainky/textswap/issues
Author-email: Kyle Cain <kyle@kylecain.me>
License-Expression: GPL-3.0-or-later
License-File: LICENSE
Keywords: bulk,cli,dictionary,find-replace,replace,text
Classifier: Development Status :: 4 - Beta
Classifier: Environment :: Console
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: GNU General Public License v3 (GPLv3)
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Topic :: Text Processing
Classifier: Topic :: Utilities
Requires-Python: >=3.9
Requires-Dist: click>=8.0.0
Provides-Extra: dev
Requires-Dist: pytest-cov>=4.0.0; extra == 'dev'
Requires-Dist: pytest>=7.0.0; extra == 'dev'
Requires-Dist: ruff>=0.1.0; extra == 'dev'
Description-Content-Type: text/markdown

# textswap

**Bulk text replacement in files using dictionary mappings.**

[![PyPI version](https://badge.fury.io/py/textswap.svg)](https://badge.fury.io/py/textswap)
[![License: GPL v3](https://img.shields.io/badge/License-GPLv3-blue.svg)](https://www.gnu.org/licenses/gpl-3.0)

Replace text across multiple files using find/replace dictionaries. Supports bidirectional replacement (keys-to-values or values-to-keys).

## Installation

```bash
pip install textswap
```

## Quick Start

1. Create a config file (`config.json`):

```json
{
  "dictionaries": {
    "example": {
      "old_text": "new_text",
      "foo": "bar"
    }
  },
  "ignore_extensions": [".exe", ".bin"],
  "ignore_directories": ["node_modules", ".git"],
  "ignore_file_prefixes": [".", "_"]
}
```

2. Run:

```bash
textswap -f ./my_folder -d 1
```

## Usage

```bash
# Interactive mode
textswap

# With options
textswap --folder ./src --direction 1 --config my_config.json

# Dry run (preview changes with diff output)
textswap -f ./src -d 1 --dry-run

# Reverse direction (values-to-keys)
textswap -f ./src -d 2
```

## Options

| Option | Short | Description |
|--------|-------|-------------|
| `--folder` | `-f` | Folder to process |
| `--direction` | `-d` | 1 = keys-to-values, 2 = values-to-keys |
| `--config` | `-c` | Path to config file (default: config.json) |
| `--dict-name` | `-n` | Dictionary name (auto-selects if only one) |
| `--dry-run` | | Preview changes without modifying files |

## Config Format

```json
{
  "dictionaries": {
    "my_replacements": {
      "find_this": "replace_with_this",
      "old": "new"
    }
  },
  "ignore_extensions": [".exe", ".dll"],
  "ignore_directories": ["node_modules", "venv"],
  "ignore_file_prefixes": [".", "_"]
}
```

## How It Works

1. **Load config**: Reads your JSON config file containing replacement dictionaries
2. **Walk directory**: Recursively traverses the target folder
3. **Filter files**: Skips files matching ignore rules (extensions, prefixes, directories)
4. **Read & replace**: For each file, reads content and applies all replacements from the dictionary
5. **Write back**: Saves modified files (or shows diff in dry-run mode)

All files are processed as UTF-8. Non-UTF-8 files are automatically skipped with a warning.

## Dry Run Output

The `--dry-run` flag shows exactly what would change without modifying files:

```bash
$ textswap -f ./src -d 1 --dry-run
Dry run mode - no files will be modified

Would modify: ./src/example.txt
--- a/./src/example.txt
+++ b/./src/example.txt
@@ -1 +1 @@
-Hello world
+Goodbye world

Processed 5 files, 1 modified
```

## Multiple Dictionaries

You can define multiple dictionaries in your config for different replacement scenarios:

```json
{
  "dictionaries": {
    "encode": {
      "secret": "s3cr3t",
      "password": "p4ssw0rd"
    },
    "localize_fr": {
      "Hello": "Bonjour",
      "Goodbye": "Au revoir"
    }
  }
}
```

Select which dictionary to use with `--dict-name`:

```bash
textswap -f ./src -d 1 -n encode
textswap -f ./src -d 1 -n localize_fr
```

## Troubleshooting

### "Invalid JSON in config file"

Your config file has a syntax error. Common issues:
- Missing commas between items
- Trailing commas (not allowed in JSON)
- Unquoted strings

Use a JSON validator to check your config.

### "Config must contain a 'dictionaries' object"

Your config file is missing the required `dictionaries` key:

```json
{
  "dictionaries": {
    "my_dict": {"find": "replace"}
  }
}
```

### Files being skipped

Files are skipped for these reasons (shown in output):
- **Not UTF-8 encoded**: Binary files or files with different encoding
- **Permission denied**: No read/write access to the file

### No files modified

Check that:
1. Your search terms exactly match the file content (case-sensitive)
2. Files aren't being filtered by ignore rules
3. The target folder contains text files

### Replacements happening in wrong order

Replacements are applied in dictionary key order. If you have overlapping patterns (e.g., "hello" and "hello world"), the first match wins. Consider using more specific patterns.

## Use Cases

- **Encoding/decoding**: Obfuscate or de-obfuscate text in files
- **Localization**: Batch replace text for different languages
- **Refactoring**: Rename variables, functions, or classes across a codebase
- **Template substitution**: Replace placeholders with actual values
- **Migration**: Update deprecated API calls or import paths

## Encoding

All files are read and written as **UTF-8**. Files that cannot be decoded as UTF-8 (binary files, files with other encodings) are automatically skipped and reported in the output.

## License

GPL v3
