-
Notifications
You must be signed in to change notification settings - Fork 423
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add a dev-tool subpackage #2167
Merged
Merged
Changes from all commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
.DS_Store | ||
/.build | ||
/Packages | ||
xcuserdata/ | ||
DerivedData/ | ||
.swiftpm/configuration/registries.json | ||
.swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata | ||
.netrc |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
// swift-tools-version:6.0 | ||
/* | ||
* Copyright 2025, gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
|
||
import PackageDescription | ||
|
||
let package = Package( | ||
name: "grpc-dev-tool", | ||
platforms: [.macOS(.v15)], | ||
dependencies: [ | ||
.package(path: "../.."), | ||
.package(url: "https://github.com/apple/swift-argument-parser.git", from: "1.5.0"), | ||
], | ||
targets: [ | ||
.executableTarget( | ||
name: "grpc-dev-tool", | ||
dependencies: [ | ||
.product(name: "GRPCCodeGen", package: "grpc-swift"), | ||
.product(name: "ArgumentParser", package: "swift-argument-parser"), | ||
] | ||
) | ||
] | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
/* | ||
* Copyright 2025, gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
|
||
import ArgumentParser | ||
|
||
@main | ||
struct GRPCDevTool: AsyncParsableCommand { | ||
static let configuration = CommandConfiguration( | ||
commandName: "grpc-dev-tool", | ||
subcommands: [GenerateJSON.self] | ||
) | ||
} |
75 changes: 75 additions & 0 deletions
75
...rpc-dev-tool/Sources/grpc-dev-tool/Subcommands/GenerateJSON/GRPCCodeGen+Conversions.swift
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
/* | ||
* Copyright 2025, gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
|
||
import GRPCCodeGen | ||
|
||
/// Creates a `ServiceDescriptor` from a JSON `ServiceSchema`. | ||
extension ServiceDescriptor { | ||
init(_ service: ServiceSchema) { | ||
self.init( | ||
documentation: "", | ||
name: .init( | ||
identifyingName: service.name, | ||
typeName: service.name, | ||
propertyName: service.name | ||
), | ||
methods: service.methods.map { | ||
MethodDescriptor($0) | ||
} | ||
) | ||
} | ||
} | ||
|
||
extension MethodDescriptor { | ||
/// Creates a `MethodDescriptor` from a JSON `ServiceSchema.Method`. | ||
init(_ method: ServiceSchema.Method) { | ||
self.init( | ||
documentation: "", | ||
name: .init( | ||
identifyingName: method.name, | ||
typeName: method.name, | ||
functionName: method.name | ||
), | ||
isInputStreaming: method.kind.streamsInput, | ||
isOutputStreaming: method.kind.streamsOutput, | ||
inputType: method.input, | ||
outputType: method.output | ||
) | ||
} | ||
} | ||
|
||
extension CodeGenerator.Config.AccessLevel { | ||
init(_ level: GeneratorConfig.AccessLevel) { | ||
switch level { | ||
case .internal: | ||
self = .internal | ||
case .package: | ||
self = .package | ||
} | ||
} | ||
} | ||
|
||
extension CodeGenerator.Config { | ||
init(_ config: GeneratorConfig) { | ||
self.init( | ||
accessLevel: CodeGenerator.Config.AccessLevel(config.accessLevel), | ||
accessLevelOnImports: config.accessLevelOnImports, | ||
client: config.generateClient, | ||
server: config.generateServer, | ||
indentation: 2 | ||
) | ||
} | ||
} |
83 changes: 83 additions & 0 deletions
83
...c-dev-tool/Sources/grpc-dev-tool/Subcommands/GenerateJSON/GRPCDevUtils+GenerateJSON.swift
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,83 @@ | ||
/* | ||
* Copyright 2025, gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
|
||
import ArgumentParser | ||
import Foundation | ||
|
||
struct GenerateJSON: ParsableCommand { | ||
static let configuration = CommandConfiguration( | ||
commandName: "generate-json", | ||
subcommands: [Generate.self, DumpConfig.self], | ||
defaultSubcommand: Generate.self | ||
) | ||
} | ||
|
||
extension GenerateJSON { | ||
struct Generate: ParsableCommand { | ||
@Argument(help: "The path to a JSON input file.") | ||
var input: String | ||
|
||
func run() throws { | ||
// Decode the input file. | ||
let url = URL(filePath: self.input) | ||
let data = try Data(contentsOf: url) | ||
let json = JSONDecoder() | ||
let config = try json.decode(JSONCodeGeneratorRequest.self, from: data) | ||
|
||
// Generate the output and dump it to stdout. | ||
let generator = JSONCodeGenerator() | ||
let sourceFile = try generator.generate(request: config) | ||
print(sourceFile.contents) | ||
} | ||
} | ||
} | ||
|
||
extension GenerateJSON { | ||
struct DumpConfig: ParsableCommand { | ||
func run() throws { | ||
// Create a request for the code generator using all four RPC kinds. | ||
var request = JSONCodeGeneratorRequest( | ||
service: ServiceSchema(name: "Echo", methods: []), | ||
config: .defaults | ||
) | ||
|
||
let methodNames = ["get", "collect", "expand", "update"] | ||
let methodKinds: [ServiceSchema.Method.Kind] = [ | ||
.unary, | ||
.clientStreaming, | ||
.serverStreaming, | ||
.bidiStreaming, | ||
] | ||
|
||
for (name, kind) in zip(methodNames, methodKinds) { | ||
let method = ServiceSchema.Method( | ||
name: name, | ||
input: "EchoRequest", | ||
output: "EchoResponse", | ||
kind: kind | ||
) | ||
request.service.methods.append(method) | ||
} | ||
|
||
// Encoding the config to JSON and dump it to stdout. | ||
let encoder = JSONEncoder() | ||
encoder.outputFormatting = [.prettyPrinted] | ||
let data = try encoder.encode(request) | ||
let json = String(decoding: data, as: UTF8.self) | ||
print(json) | ||
} | ||
} | ||
} |
119 changes: 119 additions & 0 deletions
119
dev/grpc-dev-tool/Sources/grpc-dev-tool/Subcommands/GenerateJSON/JSONCodeGenerator.swift
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
/* | ||
* Copyright 2025, gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
|
||
import Foundation | ||
import GRPCCodeGen | ||
|
||
struct JSONCodeGenerator { | ||
private static let currentYear: Int = { | ||
let now = Date() | ||
let year = Calendar.current.component(.year, from: Date()) | ||
return year | ||
}() | ||
|
||
private static let header = """ | ||
/* | ||
* Copyright \(Self.currentYear), gRPC Authors All rights reserved. | ||
* | ||
* Licensed under the Apache License, Version 2.0 (the "License"); | ||
* you may not use this file except in compliance with the License. | ||
* You may obtain a copy of the License at | ||
* | ||
* http://www.apache.org/licenses/LICENSE-2.0 | ||
* | ||
* Unless required by applicable law or agreed to in writing, software | ||
* distributed under the License is distributed on an "AS IS" BASIS, | ||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
* See the License for the specific language governing permissions and | ||
* limitations under the License. | ||
*/ | ||
""" | ||
|
||
private static let jsonSerializers: String = """ | ||
fileprivate struct JSONSerializer<Message: Codable>: MessageSerializer { | ||
fileprivate func serialize<Bytes: GRPCContiguousBytes>( | ||
_ message: Message | ||
) throws -> Bytes { | ||
do { | ||
let jsonEncoder = JSONEncoder() | ||
let data = try jsonEncoder.encode(message) | ||
return Bytes(data) | ||
} catch { | ||
throw RPCError( | ||
code: .internalError, | ||
message: "Can't serialize message to JSON.", | ||
cause: error | ||
) | ||
} | ||
} | ||
} | ||
|
||
fileprivate struct JSONDeserializer<Message: Codable>: MessageDeserializer { | ||
fileprivate func deserialize<Bytes: GRPCContiguousBytes>( | ||
_ serializedMessageBytes: Bytes | ||
) throws -> Message { | ||
do { | ||
let jsonDecoder = JSONDecoder() | ||
let data = serializedMessageBytes.withUnsafeBytes { Data($0) } | ||
return try jsonDecoder.decode(Message.self, from: data) | ||
} catch { | ||
throw RPCError( | ||
code: .internalError, | ||
message: "Can't deserialize message from JSON.", | ||
cause: error | ||
) | ||
} | ||
} | ||
} | ||
""" | ||
|
||
func generate(request: JSONCodeGeneratorRequest) throws -> SourceFile { | ||
let generator = CodeGenerator(config: CodeGenerator.Config(request.config)) | ||
|
||
let codeGenRequest = CodeGenerationRequest( | ||
fileName: request.service.name + ".swift", | ||
leadingTrivia: Self.header, | ||
dependencies: [ | ||
Dependency( | ||
item: Dependency.Item(kind: .struct, name: "Data"), | ||
module: "Foundation", | ||
accessLevel: .internal | ||
), | ||
Dependency( | ||
item: Dependency.Item(kind: .class, name: "JSONEncoder"), | ||
module: "Foundation", | ||
accessLevel: .internal | ||
), | ||
Dependency( | ||
item: Dependency.Item(kind: .class, name: "JSONDecoder"), | ||
module: "Foundation", | ||
accessLevel: .internal | ||
), | ||
], | ||
services: [ServiceDescriptor(request.service)], | ||
makeSerializerCodeSnippet: { type in "JSONSerializer<\(type)>()" }, | ||
makeDeserializerCodeSnippet: { type in "JSONDeserializer<\(type)>()" } | ||
) | ||
|
||
var sourceFile = try generator.generate(codeGenRequest) | ||
|
||
// Insert a fileprivate serializer/deserializer for JSON at the bottom of each file. | ||
sourceFile.contents += "\n\n" | ||
sourceFile.contents += Self.jsonSerializers | ||
|
||
return sourceFile | ||
} | ||
} |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we have both commands take an
output
param as well instead of printing to stdout? Not necessary but we could automate things further with a build plugin eventually if we did this instead.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Similar to the other comment, I don't think we need this now: we can just redirect the output to a file using bash etc.