cgx-core 0.0.9

Core library for cgx, the Rust equivalent of uvx or npx for running Rust crates quickly and easily
Documentation
# cgx

[![CI](https://github.com/anelson-labs/cgx/actions/workflows/ci.yml/badge.svg)](https://github.com/anelson-labs/cgx/actions/workflows/ci.yml)
[![Crates.io](https://img.shields.io/crates/v/cgx?link=https%3A%2F%2Fcrates.io%2Fcrates%2Fcgx)](https://crates.io/crates/cgx)
![license](https://img.shields.io/crates/l/cgx.svg)

Execute Rust crates easily and quickly. Like `uvx` or `npx` for Rust.

`cgx` lets you run Cargo plugins and any other Rust binaries without needing to install them first. It will do what you
would do manually with `cargo install`, `cargo binstall`, `cargo update`, and `cargo run-bin`, but in a single command.

:warning: **NOTE**: `cgx` is still under active development, and is not yet considered stable. :warning:

## Installation

### Quick Install (Recommended)

**macOS and Linux:**

```sh
curl --proto '=https' --tlsv1.2 -LsSf https://github.com/anelson-labs/cgx/releases/latest/download/cgx-installer.sh | sh
```

**Windows:**

```powershell
powershell -ExecutionPolicy ByPass -c "irm https://github.com/anelson-labs/cgx/releases/latest/download/cgx-installer.ps1 | iex"
```

The installer will download the appropriate binary for your platform and add it to your PATH.

> **Note:** To install a specific version for CI/reproducible builds, replace `latest` in the URL above with the desired version tag from the [Releases page]https://github.com/anelson-labs/cgx/releases (e.g., `v0.0.3`).

### Alternative Installation Methods

You can also install using Rust tooling:

**Via cargo install:**

```sh
cargo install cgx
```

**Via cargo-binstall (faster, uses pre-built binaries):**

```sh
cargo binstall cgx
```

**Manual download:**

Download prebuilt binaries directly from the [Releases page](https://github.com/anelson-labs/cgx/releases).

---

_Coming soon: Install via `curl https://cgx.sh/install.sh | sh` once the cgx.sh domain is set up._

## Runtime Dependencies

`cgx` uses `gix` for git operations, and for git-over-HTTP `gix` currently uses a curl-based HTTP transport backend.

Depending on your target and linkage mode, this can introduce runtime library dependencies for the `cgx` binary on
some platforms (for example Linux dynamic builds often depend on `libcurl` and `libz`).

`cgx` is configured to use the rustls-backed curl transport for git-over-HTTP, so cgx itself does not require
OpenSSL to be present as an application-level dependency.

If you run `cgx` in minimal containers or stripped-down environments, make sure those runtime libraries are present.

## Example usage

```sh
# Run ripgrep, installing it if it's missing
cgx ripgrep
```

There's a special case if the first argument is `cargo`, which indicates that you want to run a Cargo subcommand which
possibly is a third-party cargo plugin that needs to be installed.

```sh
# Run `cargo deny`, but install cargo-deny it if its missing
cgx cargo deny
```

## Argument ordering

Like `npx` and `uvx`, `cgx` requires that its own flags come **before** the crate name, and any flags intended for the executed crate come **after** the crate name:

```sh
# Correct: cgx flags before crate name, crate flags after
cgx --features serde ripgrep --color=always

# Wrong: cgx will pass --features to ripgrep as an argument
cgx ripgrep --features serde --color=always
```

You can also use `--` as an explicit separator (like `cargo run`):

```sh
# Explicit separator, equivalent to `cgx ripgrep --version` but more explicit
cgx ripgrep -- --version
```

## Version specification

The default is to use the latest version of the crate, but you can specify a version if you want, using the familiar
Rust crate version syntax:

```sh
# Run the latest release of Ripgrep with major version 14
cgx ripgrep@14

# Run the latest version of Ripgrep 14.1
cgx ripgrep@14.1
```

## Version pinning with config files

One of the handy features of tools like `uvx` and `npx` is that you can pin to a specific version of a tool in your
workspace. `cgx` supports this as well using `cgx.toml` configuration files.

Create a `cgx.toml` file in your project root:

```toml
[tools]
ripgrep = "14.1"
cargo-deny = "0.17"
```

Now, anywhere inside this directory (or its subdirectories), `cgx ripgrep` will use version 14.1, and `cgx cargo deny`
will use cargo-deny version 0.17.

You can also specify more complex configurations:

```toml
[tools]
# Simple version constraint
ripgrep = "14.1"

# Detailed configuration with features
taplo = { version = "1.0", features = ["full"] }

# Git repository source
my-tool = { git = "https://github.com/owner/repo.git", tag = "v1.0.0" }

# Custom registry
private-tool = { version = "1.0", registry = "my-registry" }

[aliases]
# Convenient short names
rg = "ripgrep"
```

### Config file hierarchy

Config files are loaded and merged in order of precedence (later sources override earlier ones):

1. System-wide config (`/etc/cgx.toml` on Linux/macOS)
2. User config (`$XDG_CONFIG_HOME/cgx/cgx.toml` or platform equivalent)
3. Directory hierarchy from filesystem root to current directory (each `cgx.toml` found)
4. Command-line arguments (highest priority)

This allows you to have global defaults in your user config while overriding them on a per-project basis.

## HTTP Configuration and Proxies

cgx makes HTTP requests to download crate metadata, pre-built binaries, and release assets from
registries, GitHub, GitLab, and other providers. These requests can be configured via the `[http]`
section in `cgx.toml`, CLI flags, or environment variables.

### Config file

```toml
[http]
timeout      = "30s"          # Request timeout (default: 30s)
retries      = 2              # Retry attempts for transient failures (default: 2)
backoff_base = "500ms"        # Base delay for exponential backoff (default: 500ms)
backoff_max  = "5s"           # Maximum delay between retries (default: 5s)
proxy        = "socks5://localhost:1080"  # HTTP or SOCKS5 proxy URL

# Proxy with basic authentication:
# proxy      = "http://user:password@proxyhost:8080"
```

### CLI flags

```sh
cgx --http-timeout 60s --http-retries 3 ripgrep
cgx --http-proxy socks5://localhost:1080 ripgrep

# Proxy with basic authentication:
cgx --http-proxy http://user:password@proxyhost:8080 ripgrep
```

### Environment variables

cgx-specific environment variables:

| Variable           | Description                         |
| ------------------ | ----------------------------------- |
| `CGX_HTTP_TIMEOUT` | Request timeout (e.g., `30s`, `2m`) |
| `CGX_HTTP_RETRIES` | Max retry count                     |
| `CGX_HTTP_PROXY`   | Proxy URL                           |

### Cargo compatibility

When no cgx-specific configuration is provided, cgx automatically honors these Cargo environment
variables:

| Cargo Variable       | cgx Equivalent   | Description                          |
| -------------------- | ---------------- | ------------------------------------ |
| `CARGO_HTTP_PROXY`   | `--http-proxy`   | HTTP/SOCKS proxy URL                 |
| `CARGO_HTTP_TIMEOUT` | `--http-timeout` | Request timeout in seconds (integer) |
| `CARGO_NET_RETRY`    | `--http-retries` | Number of retry attempts             |

The standard proxy variables `HTTPS_PROXY`, `https_proxy`, and `http_proxy` are also honored
automatically by the underlying HTTP library.

This means that if your environment is already configured for `cargo` to work behind a proxy,
`cgx` should work without any additional configuration.

For git operations (`--git`, `--github`, `--gitlab`) over HTTP/S, cgx applies the same HTTP settings where possible:
- `proxy`
- `retries` and backoff settings
- user agent
- `timeout`

For git-over-HTTP specifically, `timeout` is intentionally used both as:
- a connection timeout
- a stalled-transfer timeout threshold (via curl low-speed timeout settings)