clutch

main

Run Swift scripts that depend on packages
swift-nest/clutch

clutch 🪺 run Swift scripts from a nest of dependencies

Swift scripting is easier with clutch when

  • scripts have dependencies, or
  • you want common code in a library, or
  • you want scattered scripts checked into a common local package, or
  • scripts build & run faster because common code is pre-built in the nest package

For each script, clutch makes a matching "peer" executable in a local "nest" package:

  • A "nest" host is a Swift package with a library of the same name (e.g., in HOME/git/Build).
  • The "peer" is a copy of the script in its own executable target module in the nest package.
  • The script name.Build.swift has #!/usr/bin/env clutch on the first line and lives anywhere.
  • When the script is run, clutch creates, builds, and/or runs a peer product in the nest:
    • Build/Package.swift gets an executable product name when the script is first run.
    • Build/Sources/name/main.swift is created or updated to match the script file.
    • name.Build.swift can depend on the Build library module,
  • When everything is up-to-date, clutch runs the executable with any arguments.

Quickstart

Tested on macOS 12+ and Linux (Ubuntu LTS/latest)

Assuming ~/git, PATH has ~/bin along with swift and git (and you're ok with Sources/clutch and nests/simple/Nest)...

# Build clutch and put on PATH (for `/usr/bin/env`)
git clone https://github.com/swift-clutch/clutch.git
cd clutch && swift build 
cp .build/debug/clutch ~/bin

# Try it with a sample nest package
cp -rf nests/simple/Nest ~/git/Nest

cat > hello <<EOF
#!/usr/bin/env clutch

let you = CommandLine.arguments[1...].first ?? "World"
print("Hello \(you)")
EOF

chmod +x hello 
./hello friend    # builds, runs `~/git/Nest/Sources/hello/main.swift`

# Use clutch directly to run or manage peers or nests
clutch hello      # run by name from anywhere
clutch hello.Nest # specify nest if not the default

clutch cat-hello  # output peer source (`clutch cat-start > newscript.swift`)
clutch path-hello # echo peer path (`vi "$(clutch path-hello)"`)
clutch peers-Nest # list peers known in nest `Nest`
clutch dir-Nest   # emit location of nest `Nest`

Usage

Install clutch, create a nest package, and write a Swift script file anywhere.

When you invoke the script, clutch runs it after ensuring its nest package peer (Nest/Sources/peer/peer.swift) is created, updated, and/or built. A nest package has a nest library (with the nest name) for common code and dependencies.

Write script: #!/usr/bin/env clutch

  • The peer name is the initial filename segment (before .).
    • The nest name is any trailing segment (ignoring .swift), or Nest.
    • Both the nest and peer name must be valid ASCII identifiers.
  • The file is executable and has a valid hash-bang on the first line:
    • #!/path/to/clutch
    • #!/usr/bin/env clutch (best, if clutch is on your PATH)
  • The script has valid top-level code, depending only on the nest library.

The nest peer in {nest}/Sources/{peer} will be created on first impression. The peer filename is main.swift, or {peer}.swift if it contains @main.

Package.swift will be updated with peer product and target declarations:

  • .executable(name: "{peer}", targets: [ "{peer}" ]),
  • .executableTarget(name: "{peer}", dependencies: ["{nest}"]),

Build in nest: $HOME/git/{nest}/Package.swift

  products: [
    // peer product created for each script, using the script name {peer}
    .executable(name: "{peer}", targets: [ "{peer}" ]),
    .library(name: "{nest}", targets: ["{nest}"]),
  ],
  targets: [
    // peer executable created for each script
    .executableTarget(name: "{peer}", dependencies: ["{nest}"]),
    .target(
      name: "{nest}",
      dependencies: [ ... ] 

By default clutch builds using -c debug --quiet (to avoid delay and noise), the nest package is named Nest, and it lives at $HOME/git/Nest.

To configure the nest location or output, set environment variables:

  • CLUTCH_NEST_NAME: find nest in $HOME/{relative-path}/{nest-name}
  • CLUTCH_NEST_RELPATH: relative path from HOME (defaults to git)
  • CLUTCH_NEST_BASE: find nest in $CLUTCH_NEST_BASE/{nest-name} instead
  • CLUTCH_NEST_PATH: full path to nest directory (ignoring other variables)
  • CLUTCH_LOG: any value to log steps to standard error
  • CLUTCH_BUILD: @{arg0}@{arg1}.., or {release} {loud | verbose}

The nest directory name must be the name of the library module.

For sample nest packages, see nests or use swift package init --type library.

Using clutch directly to run and manage peers

clutch name{.Nest}             # Run peer by name
clutch [peers|path]-Nest       # Emit Nest peers or location
clutch [cat|path]-name{.Nest}  # Emit peer code or location

Use clutch directly to run scripts by filename or peer name

clutch name.swift      # Build and run name from default nest (even if new)
clutch name            # Run by name

Use clutch to list peers in a nest and find or copy the peer source file:

clutch peers-Data      # List peers in the `Data` nest
clutch path-name       # Echo path to source file for peer `name`
clutch cat-init.Data   # Output code from peer `init` in Data nest

Limitations (under development)

  • For new scripts, @main and Package.swift operations fail on unexpected input.
  • swift build keeps the old executable when updates produce no binary changes.
  • For known bugs and missing features, see README-clutch.
  • The @main detection is simplistic for new scripts (and not done for updates).
  • The Package.swift editing for new scripts is also a simple scan.
    • It seeks products: and targets: (the latter with 2 leading spaces)
      • target: is common; please avoid 2 spaces before it.
      • And please avoid that text in comments or other declarations.
    • To avoid missed/invalid insertions, tag the line before the declaration: with CLUTCH_PRODUCT or CLUTCH_TARGET
  • Builds are based only on last-modified time.
    • Swift does not re-link the binary after edits result in the same code (so a second clutch run would trigger another no-op build).
  • Output streams and exit codes mix clutch, swift build, and executables.
    • A failed exit code will always be 1 (even if the script exits with 2).
  • Users have to manually edit the package to rename/delete peers or fix errors.
    • To delete, remove Sources/peer and two lines for peer in Package.swift

Package Status

  • Tested on macOS/Linux, but unproven in the wild...
  • Command set, CLI interface, configuration, and error-handling could change.

Alternatives and related libraries

  • clatch is clutch done quick, just for scripts.
    • No commands, no environment configuration, no error or SystemCalls wrapper
  • The swift command works fine if no libraries are needed.
    • Use #!/usr/bin/env swift at the start of a script file to run directly
    • swift script.swift does the same, without the #! hash-bang line
    • swift -e 'statement{; statement}' runs a snippet of code
    • Or generateCode | swift - to run code from the input stream
  • swift-sh builds and runs using libraries from import comments
  • Try Swift Argument Parser to simplify writing CLI's
  • Try Shwift for async cross-platform scripting
  • To install Swift package command-line tools, consider Mint
    • mint install swift-nest/clutch

Development

  • Please create an issue with any feedback, to help get to 1.0 :)
  • See README-clutch for known issues
  • License: Apache 2.0
  • Copyright Contributors on their contribution date. All rights reserved.

Description

  • Swift Tools 5.5.0
View More Packages from this Author

Dependencies

Last updated: Thu Oct 31 2024 07:10:19 GMT-0900 (Hawaii-Aleutian Daylight Time)