Metadata-Version: 2.4
Name: ftp-winmount
Version: 0.1.1
Summary: Mount FTP server as a local Windows drive using WinFsp
Author: Daniel T Sasser II
License-Expression: MIT
Project-URL: Homepage, https://github.com/dansasser/ftp-winmount
Project-URL: Documentation, https://github.com/dansasser/ftp-winmount#readme
Project-URL: Repository, https://github.com/dansasser/ftp-winmount.git
Project-URL: Issues, https://github.com/dansasser/ftp-winmount/issues
Keywords: ftp,mount,windows,drive,winfsp,filesystem
Classifier: Development Status :: 4 - Beta
Classifier: Environment :: Console
Classifier: Environment :: Win32 (MS Windows)
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: End Users/Desktop
Classifier: Intended Audience :: System Administrators
Classifier: Operating System :: Microsoft :: Windows
Classifier: Operating System :: Microsoft :: Windows :: Windows 10
Classifier: Operating System :: Microsoft :: Windows :: Windows 11
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Topic :: Internet :: File Transfer Protocol (FTP)
Classifier: Topic :: System :: Filesystems
Requires-Python: >=3.10
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: winfspy>=0.8.0; platform_system == "Windows"
Requires-Dist: cachetools>=5.0.0
Provides-Extra: dev
Requires-Dist: pytest>=7.0.0; extra == "dev"
Requires-Dist: pytest-cov>=4.0.0; extra == "dev"
Requires-Dist: pyftpdlib>=1.5.0; extra == "dev"
Requires-Dist: psutil>=5.9.0; extra == "dev"
Requires-Dist: ruff>=0.1.0; extra == "dev"
Requires-Dist: mypy>=1.0.0; extra == "dev"
Requires-Dist: build>=1.0.0; extra == "dev"
Requires-Dist: twine>=4.0.0; extra == "dev"
Requires-Dist: pyinstaller>=6.0.0; extra == "dev"
Dynamic: license-file
Dynamic: requires-python

# FTP-WinMount

[![PyPI version](https://badge.fury.io/py/ftp-winmount.svg)](https://badge.fury.io/py/ftp-winmount)
[![CI](https://github.com/dansasser/ftp-winmount/actions/workflows/ci.yml/badge.svg)](https://github.com/dansasser/ftp-winmount/actions/workflows/ci.yml)
[![Python 3.10+](https://img.shields.io/badge/python-3.10+-blue.svg)](https://www.python.org/downloads/)
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
[![Windows](https://img.shields.io/badge/platform-Windows-blue.svg)](https://www.microsoft.com/windows)

Mount any FTP server as a local Windows drive. No sync, no ads, no telemetry.

---

## Download

**No Python required.** Download from [GitHub Releases](https://github.com/dansasser/ftp-winmount/releases/latest).

### Installer (Recommended)

1. Install [WinFsp](https://winfsp.dev/rel/) (required dependency)
2. Download `ftp-winmount-X.X.X-setup.exe`
3. Run the installer (installs to Program Files and adds to PATH)
4. Open any terminal and run:
   ```bash
   ftp-winmount mount --host 192.168.0.130 --port 2121 --drive Z
   ```

### Portable

If you don't want to install:

1. Install [WinFsp](https://winfsp.dev/rel/) (required dependency)
2. Download `ftp-winmount-X.X.X-portable.exe`
3. Run from wherever you saved it:
   ```bash
   C:\Downloads\ftp-winmount-0.1.0-portable.exe mount --host 192.168.0.130 --port 2121 --drive Z
   ```

---

## Why?

- VS Code's FTP plugins only sync files, they don't mount
- Commercial tools like RaiDrive and NetDrive come with ads and tracking
- Windows can't natively mount FTP as a drive letter
- You want to own your tools

---

## Features

- Mount FTP server as a Windows drive letter (Z:, Y:, etc.)
- Full read/write support
- Works with any application (VS Code, File Explorer, Notepad, etc.)
- Anonymous or authenticated FTP
- Auto-reconnect on connection drop
- Lightweight, no background services when not in use
- Open source, no ads, no tracking

---

## Requirements

### Windows Dependencies

FTP-WinMount requires **WinFsp** (Windows File System Proxy) - a free, open-source file system driver that enables user-mode filesystems on Windows.

| Dependency | Version | Required | Download |
|------------|---------|----------|----------|
| Windows | 10 or 11 | Yes | - |
| Python | 3.10+ | Yes | [python.org](https://www.python.org/downloads/) |
| WinFsp | 2.0+ | Yes | [winfsp.dev/rel](https://winfsp.dev/rel/) |

### Installing WinFsp

**Option 1: Download installer (Recommended)**
1. Go to https://winfsp.dev/rel/
2. Download the latest `.msi` installer
3. Run the installer (requires admin rights)
4. Restart any open terminals

**Option 2: Winget**
```powershell
winget install WinFsp.WinFsp
```

**Option 3: Chocolatey**
```powershell
choco install winfsp
```

**Verify Installation:**
```powershell
# Check WinFsp is installed
dir "C:\Program Files (x86)\WinFsp"
```

---

## Installation

### Installer (Easiest)

See [Download](#download) section above. No Python required. Installs to Program Files and adds to PATH.

### From PyPI

```bash
pip install ftp-winmount
```

This installs `ftp-winmount` to your Python Scripts folder. Make sure that folder is in your PATH.

**Note:** If you install inside a virtual environment, the command will only work when that venv is activated.

### From Source

```bash
git clone https://github.com/dansasser/ftp-winmount.git
cd ftp-winmount
pip install .
```

### Development Install

```bash
git clone https://github.com/dansasser/ftp-winmount.git
cd ftp-winmount
pip install -e ".[dev]"
```

### Building Standalone Executable

To build `ftp-winmount.exe` yourself:

```bash
git clone https://github.com/dansasser/ftp-winmount.git
cd ftp-winmount
pip install -e ".[dev]"
python build_exe.py
```

The executable will be at `dist/ftp-winmount.exe`.

### Building Installer

To build the Windows installer (requires [Inno Setup](https://jrsoftware.org/isinfo.php)):

```bash
python build_exe.py
iscc installer.iss
```

The installer will be at `dist/ftp-winmount-X.X.X-setup.exe`.

---

## Quick Start

Mount an anonymous FTP server:
```bash
ftp-winmount mount --host 192.168.0.130 --port 2121 --drive Z
```

Your FTP server is now accessible at `Z:\`

To unmount:
```bash
ftp-winmount unmount --drive Z
```

Or just press `Ctrl+C` in the terminal.

---

## Usage

### Command Line
```bash
# Mount with minimal options
ftp-winmount mount --host 192.168.0.130 --drive Z

# Mount with custom port
ftp-winmount mount --host 192.168.0.130 --port 2121 --drive Z

# Mount with authentication
ftp-winmount mount --host ftp.example.com --user myuser --password mypass --drive Z

# Mount with config file
ftp-winmount mount --config config.ini

# Check status
ftp-winmount status

# Unmount
ftp-winmount unmount --drive Z
```

### Configuration File

Create `config.ini`:
```ini
[ftp]
host = 192.168.0.130
port = 2121
username =
password =

[mount]
drive_letter = Z
volume_label = My FTP Drive

[cache]
directory_ttl_seconds = 30

[logging]
level = INFO
file = ftp-winmount.log
```

Then run:
```bash
ftp-winmount mount --config config.ini
```

---

## Use with VS Code

1. Mount your FTP server:
```bash
   ftp-winmount mount --host 192.168.0.130 --port 2121 --drive Z
```

2. Open VS Code

3. File -> Open Folder -> Select `Z:\`

4. Edit files normally. Changes save directly to the FTP server.

---

## Use with Python's pyftpdlib

If you're running a simple FTP server with pyftpdlib:

**Server (host machine):**
```bash
pip install pyftpdlib
cd /path/to/your/files
python -m pyftpdlib -p 2121 -i 0.0.0.0 -w
```

**Client (your machine):**
```bash
ftp-winmount mount --host SERVER_IP --port 2121 --drive Z
```

---

## Options

| Option | Description | Default |
|--------|-------------|---------|
| `--host` | FTP server hostname or IP | Required |
| `--port` | FTP server port | 21 |
| `--drive` | Drive letter to mount | Required |
| `--user` | FTP username | Anonymous |
| `--password` | FTP password | None |
| `--config` | Path to config file | None |
| `--verbose` | Enable debug logging | False |

---

## Troubleshooting

### "WinFsp not found" or "winfspy not available"

WinFsp must be installed system-wide. The Python package `winfspy` is just bindings.

1. Download WinFsp from https://winfsp.dev/rel/
2. Run the installer as administrator
3. Restart your terminal
4. Try again

### "Drive letter in use"

Choose a different drive letter or unmount the existing drive:
```bash
net use Z: /delete
```

### "Connection refused"

- Check the FTP server is running
- Check the IP address and port are correct
- Check firewall isn't blocking the connection

### "Permission denied"

- For anonymous FTP, ensure the server allows anonymous access
- For authenticated FTP, check username and password

### Files appear but can't be read

- FTP server may have disconnected
- Try unmounting and remounting

### Slow performance

- Enable caching in config file
- Increase `directory_ttl_seconds`
- Check network connection to FTP server

### Mount crashes or hangs

- Update to the latest version of WinFsp
- Check Windows Event Viewer for errors
- Run with `--verbose` flag for detailed logs

---

## Limitations

- **Windows only** - This tool uses WinFsp which is Windows-specific
- FTP protocol does not support file locking. Concurrent writes from multiple clients may cause issues.
- Some FTP servers don't report accurate file sizes or timestamps.
- Very large files (>1GB) may be slow due to FTP protocol limitations.

---

## Development

### Setup

```bash
git clone https://github.com/dansasser/ftp-winmount.git
cd ftp-winmount
python -m venv venv
venv\Scripts\activate
pip install -e ".[dev]"
```

### Run Tests

```bash
pytest
```

### Linting

```bash
ruff check ftp_winmount tests
ruff format ftp_winmount tests
```

### Build Package

```bash
python -m build
twine check dist/*
```

---

## Contributing

Contributions welcome. Please open an issue first to discuss major changes.

1. Fork the repository
2. Create a feature branch (`git checkout -b feature/amazing-feature`)
3. Commit your changes (`git commit -m 'Add amazing feature'`)
4. Push to the branch (`git push origin feature/amazing-feature`)
5. Open a Pull Request

---

## License

MIT License. See [LICENSE](LICENSE) file.

---

## Acknowledgments

- [WinFsp](https://winfsp.dev/) - Windows File System Proxy
- [winfspy](https://github.com/Scille/winfspy) - Python bindings for WinFsp

---

## Related Projects

- [WinFsp](https://winfsp.dev/) - The underlying filesystem driver
- [SSHFS-Win](https://github.com/winfsp/sshfs-win) - Mount SSH/SFTP as drives (uses WinFsp)
- [rclone](https://rclone.org/) - Mount cloud storage (supports many providers)
