Skip to content

Commit

Permalink
Add merge function to command groups (#195)
Browse files Browse the repository at this point in the history
* add merge function to command groups

* Fix compilation error

---------

Co-authored-by: Tim <[email protected]>
  • Loading branch information
marius-se and 0xTim authored Dec 1, 2023
1 parent 39f219a commit 18262d2
Show file tree
Hide file tree
Showing 6 changed files with 63 additions and 14 deletions.
9 changes: 6 additions & 3 deletions Sources/ConsoleKitCommands/Async/AsyncCommand.swift
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ import ConsoleKitTerminal
/// U ||----w |
/// || ||
///
public protocol AsyncCommand: AnyAsyncCommand {
public protocol AsyncCommand: Sendable, AnyAsyncCommand {
associatedtype Signature: CommandSignature
func run(using context: CommandContext, signature: Signature) async throws
}
Expand All @@ -91,7 +91,7 @@ extension AsyncCommand {
guard context.input.arguments.isEmpty else {
throw CommandError.unknownInput(context.input.arguments.joined(separator: " "))
}
try await self.run(using: context, signature: signature)
try await self.run(using: context, signature: signature)
}

public func outputAutoComplete(using context: inout CommandContext) {
Expand All @@ -102,7 +102,10 @@ extension AsyncCommand {
}

public func outputHelp(using context: inout CommandContext) {
context.console.output("Usage: ".consoleText(.info) + context.input.executable.consoleText() + " ", newLine: false)
context.console.output(
"Usage: ".consoleText(.info) + context.input.executable.consoleText() + " ",
newLine: false
)
Signature().outputHelp(help: self.help, using: &context)
}
}
34 changes: 32 additions & 2 deletions Sources/ConsoleKitCommands/Async/AsyncCommandGroup.swift
Original file line number Diff line number Diff line change
Expand Up @@ -18,11 +18,41 @@ import ConsoleKitTerminal
public protocol AsyncCommandGroup: AnyAsyncCommand {
var commands: [String: any AnyAsyncCommand] { get }
var defaultCommand: (any AnyAsyncCommand)? { get }

/// Merges this group with another group.
/// - Parameters:
/// - group: The group to merge with.
/// - defaultCommand: The new default command to use.
/// - help: The help message to use for the merged group.
/// - Returns: A new `AsyncCommandGroup` with the merged commands.
func merge(
with group: any AsyncCommandGroup,
defaultCommand: (any AnyAsyncCommand)?,
help: String
) -> any AsyncCommandGroup
}

public struct MergedAsyncCommandGroup: AsyncCommandGroup {
public let commands: [String: any AnyAsyncCommand]
public let defaultCommand: (any AnyAsyncCommand)?
public var help: String
}

extension AsyncCommandGroup {
public var defaultCommand: (any AnyAsyncCommand)? {
nil
public var defaultCommand: (any AnyAsyncCommand)? { nil }

public func merge(
with group: any AsyncCommandGroup,
defaultCommand: (any AnyAsyncCommand)?,
help: String
) -> any AsyncCommandGroup {
var mergedCommands = self.commands
mergedCommands.merge(group.commands, uniquingKeysWith: { (_, new) in new })
return MergedAsyncCommandGroup(
commands: mergedCommands,
defaultCommand: defaultCommand,
help: help
)
}
}

Expand Down
4 changes: 2 additions & 2 deletions Sources/ConsoleKitCommands/Async/AsyncCommands.swift
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/// Represents a top-level group of configured commands. This is usually created by calling `resolve(for:)` on `AsyncCommands`.
public struct AsyncCommands {
public struct AsyncCommands: Sendable {
/// Top-level available commands, stored by unique name.
public var commands: [String: any AnyAsyncCommand]

Expand Down Expand Up @@ -38,7 +38,7 @@ public struct AsyncCommands {
self.defaultCommand = defaultCommand
self.enableAutocomplete = enableAutocomplete
}

/// Adds an `AsyncCommand` instance to the config.
///
/// var config = AsyncCommands()
Expand Down
13 changes: 11 additions & 2 deletions Sources/ConsoleKitCommands/Base/AnyCommand.swift
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
/// A type-erased `Command`.
public protocol AnyCommand: Sendable {
public protocol AnyCommand: Sendable, AnyAsyncCommand {
/// Text that will be displayed when `--help` is passed.
var help: String { get }

/// Runs the command against the supplied input.
func run(using context: inout CommandContext) throws
func outputAutoComplete(using context: inout CommandContext) throws
Expand All @@ -24,4 +24,13 @@ extension AnyCommand {
public func renderCompletionFunctions(using context: CommandContext, shell: Shell) -> String {
""
}

// we need to have a sync environment so the compiler uses the sync run method over the async version
private func syncRun(using context: inout CommandContext) throws {
try self.run(using: &context)
}

public func run(using context: inout CommandContext) async throws {
try self.syncRun(using: &context)
}
}
9 changes: 5 additions & 4 deletions Sources/ConsoleKitCommands/Base/CommandContext.swift
Original file line number Diff line number Diff line change
@@ -1,15 +1,16 @@
import protocol ConsoleKitTerminal.Console
import struct ConsoleKitTerminal.AnySendableHashable

/// A type-erased `CommandContext`
public struct CommandContext {
public struct CommandContext: Sendable {
/// The `Console` this command was run on.
public var console: any Console

/// The parsed arguments (according to declared signature).
public var input: CommandInput

public var userInfo: [AnyHashable: Any]
public var userInfo: [AnySendableHashable: any Sendable]

/// Create a new `AnyCommandContext`.
public init(
console: any Console,
Expand Down
8 changes: 7 additions & 1 deletion Sources/ConsoleKitCommands/Base/CommandGroup.swift
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import ConsoleKitTerminal
/// try console.run(group, with: context)
///
/// You can create your own `CommandGroup` if you want to support custom `CommandOptions`.
public protocol CommandGroup: AnyCommand {
public protocol CommandGroup: AnyCommand, AsyncCommandGroup {
var commands: [String: any AnyCommand] { get }
var defaultCommand: (any AnyCommand)? { get }
}
Expand All @@ -24,6 +24,12 @@ extension CommandGroup {
public var defaultCommand: (any AnyCommand)? {
nil
}

public var commands: [String: any AnyAsyncCommand] {
// make the compiler happy
let castedCommands: [String: any AnyCommand] = commands
return castedCommands
}
}

extension CommandGroup {
Expand Down

0 comments on commit 18262d2

Please sign in to comment.