Skip to content

Commit ad7f0e7

Browse files
Idiomatic naming strategy as opt-in (#679)
### Motivation Implementation of #683. ### Modifications - Implemented the SOAR-0013 idiomatic naming strategy - Added name overrides - Switched most reference tests to use the new strategy - Updated some docs (I'll update the rest after this lands and gets released, then I can update Examples and IntegrationTest) ### Result SOAR-0013 implemented. ### Test Plan Updated and added new unit tests. --------- Co-authored-by: Si Beaumont <[email protected]> Co-authored-by: Si Beaumont <[email protected]>
1 parent 4d0c02d commit ad7f0e7

File tree

61 files changed

+1263
-727
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

61 files changed

+1263
-727
lines changed

Sources/_OpenAPIGeneratorCore/Config.swift

+30
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,20 @@
1212
//
1313
//===----------------------------------------------------------------------===//
1414

15+
/// A strategy for turning OpenAPI identifiers into Swift identifiers.
16+
public enum NamingStrategy: String, Sendable, Codable, Equatable {
17+
18+
/// A defensive strategy that can handle any OpenAPI identifier and produce a non-conflicting Swift identifier.
19+
///
20+
/// Introduced in [SOAR-0001](https://swiftpackageindex.com/apple/swift-openapi-generator/documentation/swift-openapi-generator/soar-0001).
21+
case defensive
22+
23+
/// An idiomatic strategy that produces Swift identifiers that more likely conform to Swift conventions.
24+
///
25+
/// Introduced in [SOAR-0013](https://swiftpackageindex.com/apple/swift-openapi-generator/documentation/swift-openapi-generator/soar-0013).
26+
case idiomatic
27+
}
28+
1529
/// A structure that contains configuration options for a single execution
1630
/// of the generator pipeline run.
1731
///
@@ -35,6 +49,14 @@ public struct Config: Sendable {
3549
/// Filter to apply to the OpenAPI document before generation.
3650
public var filter: DocumentFilter?
3751

52+
/// The naming strategy to use for deriving Swift identifiers from OpenAPI identifiers.
53+
///
54+
/// Defaults to `defensive`.
55+
public var namingStrategy: NamingStrategy
56+
57+
/// A map of OpenAPI identifiers to desired Swift identifiers, used instead of the naming strategy.
58+
public var nameOverrides: [String: String]
59+
3860
/// Additional pre-release features to enable.
3961
public var featureFlags: FeatureFlags
4062

@@ -44,18 +66,26 @@ public struct Config: Sendable {
4466
/// - access: The access modifier to use for generated declarations.
4567
/// - additionalImports: Additional imports to add to each generated file.
4668
/// - filter: Filter to apply to the OpenAPI document before generation.
69+
/// - namingStrategy: The naming strategy to use for deriving Swift identifiers from OpenAPI identifiers.
70+
/// Defaults to `defensive`.
71+
/// - nameOverrides: A map of OpenAPI identifiers to desired Swift identifiers, used instead
72+
/// of the naming strategy.
4773
/// - featureFlags: Additional pre-release features to enable.
4874
public init(
4975
mode: GeneratorMode,
5076
access: AccessModifier,
5177
additionalImports: [String] = [],
5278
filter: DocumentFilter? = nil,
79+
namingStrategy: NamingStrategy = .defensive,
80+
nameOverrides: [String: String] = [:],
5381
featureFlags: FeatureFlags = []
5482
) {
5583
self.mode = mode
5684
self.access = access
5785
self.additionalImports = additionalImports
5886
self.filter = filter
87+
self.namingStrategy = namingStrategy
88+
self.nameOverrides = nameOverrides
5989
self.featureFlags = featureFlags
6090
}
6191
}

Sources/_OpenAPIGeneratorCore/Translator/ClientTranslator/translateClientMethod.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -145,7 +145,7 @@ extension ClientFileTranslator {
145145
func translateClientMethod(_ description: OperationDescription) throws -> Declaration {
146146

147147
let operationTypeExpr = Expression.identifierType(.member(Constants.Operations.namespace))
148-
.dot(description.methodName)
148+
.dot(description.operationTypeName)
149149

150150
let operationArg = FunctionArgumentDescription(label: "forOperation", expression: operationTypeExpr.dot("id"))
151151
let inputArg = FunctionArgumentDescription(

Sources/_OpenAPIGeneratorCore/Translator/CommonTranslations/SwiftSafeNames.swift

-89
This file was deleted.

Sources/_OpenAPIGeneratorCore/Translator/CommonTranslations/translateAllAnyOneOf.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,7 @@ extension TypesFileTranslator {
144144
userDescription: nil,
145145
parent: typeName
146146
)
147-
let caseName = safeSwiftNameForOneOfMappedType(mappedType)
147+
let caseName = safeSwiftNameForOneOfMappedCase(mappedType)
148148
return (caseName, mappedType.rawNames, true, comment, mappedType.typeName.asUsage, [])
149149
}
150150
} else {
@@ -209,7 +209,7 @@ extension TypesFileTranslator {
209209
let decoder: Declaration
210210
if let discriminator {
211211
let originalName = discriminator.propertyName
212-
let swiftName = context.asSwiftSafeName(originalName)
212+
let swiftName = context.safeNameGenerator.swiftMemberName(for: originalName)
213213
codingKeysDecls = [
214214
.enum(
215215
accessModifier: config.access,

Sources/_OpenAPIGeneratorCore/Translator/CommonTranslations/translateRawEnum.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -101,12 +101,12 @@ extension FileTranslator {
101101
// This is unlikely to be fixed, so handling that case here.
102102
// https://github.com/apple/swift-openapi-generator/issues/118
103103
if isNullable && anyValue is Void {
104-
try addIfUnique(id: .string(""), caseName: context.asSwiftSafeName(""))
104+
try addIfUnique(id: .string(""), caseName: context.safeNameGenerator.swiftMemberName(for: ""))
105105
} else {
106106
guard let rawValue = anyValue as? String else {
107107
throw GenericError(message: "Disallowed value for a string enum '\(typeName)': \(anyValue)")
108108
}
109-
let caseName = context.asSwiftSafeName(rawValue)
109+
let caseName = context.safeNameGenerator.swiftMemberName(for: rawValue)
110110
try addIfUnique(id: .string(rawValue), caseName: caseName)
111111
}
112112
case .integer:

Sources/_OpenAPIGeneratorCore/Translator/CommonTypes/DiscriminatorExtensions.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -79,8 +79,8 @@ extension FileTranslator {
7979
/// component.
8080
/// - Parameter type: The `OneOfMappedType` for which to determine the case name.
8181
/// - Returns: A string representing the safe Swift name for the specified `OneOfMappedType`.
82-
func safeSwiftNameForOneOfMappedType(_ type: OneOfMappedType) -> String {
83-
context.asSwiftSafeName(type.rawNames[0])
82+
func safeSwiftNameForOneOfMappedCase(_ type: OneOfMappedType) -> String {
83+
context.safeNameGenerator.swiftMemberName(for: type.rawNames[0])
8484
}
8585
}
8686

Sources/_OpenAPIGeneratorCore/Translator/CommonTypes/StructBlueprint.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -153,7 +153,7 @@ struct PropertyBlueprint {
153153
extension PropertyBlueprint {
154154

155155
/// A name that is verified to be a valid Swift identifier.
156-
var swiftSafeName: String { context.asSwiftSafeName(originalName) }
156+
var swiftSafeName: String { context.safeNameGenerator.swiftMemberName(for: originalName) }
157157

158158
/// The JSON path to the property.
159159
///

Sources/_OpenAPIGeneratorCore/Translator/FileTranslator.swift

+38-6
Original file line numberDiff line numberDiff line change
@@ -44,18 +44,50 @@ protocol FileTranslator {
4444
func translateFile(parsedOpenAPI: ParsedOpenAPIRepresentation) throws -> StructuredSwiftRepresentation
4545
}
4646

47+
/// A generator that allows overriding the documented name.
48+
struct OverridableSafeNameGenerator: SafeNameGenerator {
49+
50+
/// The upstream name generator for names that aren't overriden.
51+
var upstream: any SafeNameGenerator
52+
53+
/// A set of overrides, where the key is the documented name and the value the desired identifier.
54+
var overrides: [String: String]
55+
56+
func swiftTypeName(for documentedName: String) -> String {
57+
if let override = overrides[documentedName] { return override }
58+
return upstream.swiftTypeName(for: documentedName)
59+
}
60+
61+
func swiftMemberName(for documentedName: String) -> String {
62+
if let override = overrides[documentedName] { return override }
63+
return upstream.swiftMemberName(for: documentedName)
64+
}
65+
66+
func swiftContentTypeName(for contentType: ContentType) -> String {
67+
upstream.swiftContentTypeName(for: contentType)
68+
}
69+
}
70+
4771
extension FileTranslator {
4872

4973
/// A new context from the file translator.
50-
var context: TranslatorContext { TranslatorContext(asSwiftSafeName: { $0.safeForSwiftCode }) }
74+
var context: TranslatorContext {
75+
let safeNameGenerator: any SafeNameGenerator
76+
switch config.namingStrategy {
77+
case .defensive: safeNameGenerator = .defensive
78+
case .idiomatic: safeNameGenerator = .idiomatic
79+
}
80+
let overridingGenerator = OverridableSafeNameGenerator(
81+
upstream: safeNameGenerator,
82+
overrides: config.nameOverrides
83+
)
84+
return TranslatorContext(safeNameGenerator: overridingGenerator)
85+
}
5186
}
5287

5388
/// A set of configuration values for concrete file translators.
5489
struct TranslatorContext {
5590

56-
/// A closure that returns a copy of the string modified to be a valid Swift identifier.
57-
///
58-
/// - Parameter string: The string to convert to be safe for Swift.
59-
/// - Returns: A Swift-safe version of the input string.
60-
var asSwiftSafeName: (String) -> String
91+
/// A type that generates safe names for use as Swift identifiers.
92+
var safeNameGenerator: any SafeNameGenerator
6193
}

Sources/_OpenAPIGeneratorCore/Translator/Multipart/MultipartContentInspector.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -120,7 +120,7 @@ extension FileTranslator {
120120
}
121121
var parts: [MultipartSchemaTypedContent] = try topLevelObject.properties.compactMap {
122122
(key, value) -> MultipartSchemaTypedContent? in
123-
let swiftSafeName = context.asSwiftSafeName(key)
123+
let swiftSafeName = context.safeNameGenerator.swiftTypeName(for: key)
124124
let typeName = typeName.appending(
125125
swiftComponent: swiftSafeName + Constants.Global.inlineTypeSuffix,
126126
jsonComponent: key

Sources/_OpenAPIGeneratorCore/Translator/Multipart/translateMultipart.swift

+3-3
Original file line numberDiff line numberDiff line change
@@ -137,7 +137,7 @@ extension TypesFileTranslator {
137137
switch part {
138138
case .documentedTyped(let documentedPart):
139139
let caseDecl: Declaration = .enumCase(
140-
name: context.asSwiftSafeName(documentedPart.originalName),
140+
name: context.safeNameGenerator.swiftMemberName(for: documentedPart.originalName),
141141
kind: .nameWithAssociatedValues([.init(type: .init(part.wrapperTypeUsage))])
142142
)
143143
let decl = try translateMultipartPartContent(
@@ -404,7 +404,7 @@ extension FileTranslator {
404404
switch part {
405405
case .documentedTyped(let part):
406406
let originalName = part.originalName
407-
let identifier = context.asSwiftSafeName(originalName)
407+
let identifier = context.safeNameGenerator.swiftMemberName(for: originalName)
408408
let contentType = part.partInfo.contentType
409409
let partTypeName = part.typeName
410410
let schema = part.schema
@@ -613,7 +613,7 @@ extension FileTranslator {
613613
switch part {
614614
case .documentedTyped(let part):
615615
let originalName = part.originalName
616-
let identifier = context.asSwiftSafeName(originalName)
616+
let identifier = context.safeNameGenerator.swiftMemberName(for: originalName)
617617
let contentType = part.partInfo.contentType
618618
let headersTypeName = part.typeName.appending(
619619
swiftComponent: Constants.Operation.Output.Payload.Headers.typeName,

Sources/_OpenAPIGeneratorCore/Translator/Operations/OperationDescription.swift

+10-3
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,14 @@ extension OperationDescription {
8383
/// Uses the `operationID` value in the OpenAPI operation, if one was
8484
/// specified. Otherwise, computes a unique name from the operation's
8585
/// path and HTTP method.
86-
var methodName: String { context.asSwiftSafeName(operationID) }
86+
var methodName: String { context.safeNameGenerator.swiftMemberName(for: operationID) }
87+
88+
/// Returns a Swift-safe type name for the operation.
89+
///
90+
/// Uses the `operationID` value in the OpenAPI operation, if one was
91+
/// specified. Otherwise, computes a unique name from the operation's
92+
/// path and HTTP method.
93+
var operationTypeName: String { context.safeNameGenerator.swiftTypeName(for: operationID) }
8794

8895
/// Returns the identifier for the operation.
8996
///
@@ -103,7 +110,7 @@ extension OperationDescription {
103110
.init(
104111
components: [.root, .init(swift: Constants.Operations.namespace, json: "paths")]
105112
+ path.components.map { .init(swift: nil, json: $0) } + [
106-
.init(swift: methodName, json: httpMethod.rawValue)
113+
.init(swift: operationTypeName, json: httpMethod.rawValue)
107114
]
108115
)
109116
}
@@ -292,7 +299,7 @@ extension OperationDescription {
292299
}
293300
let newPath = OpenAPI.Path(newComponents, trailingSlash: path.trailingSlash)
294301
let names: [Expression] = orderedPathParameters.map { param in
295-
.identifierPattern("input").dot("path").dot(context.asSwiftSafeName(param))
302+
.identifierPattern("input").dot("path").dot(context.safeNameGenerator.swiftMemberName(for: param))
296303
}
297304
let arrayExpr: Expression = .literal(.array(names))
298305
return (newPath.rawValue, arrayExpr)

Sources/_OpenAPIGeneratorCore/Translator/Parameters/TypedParameter.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,7 @@ extension TypedParameter {
4848
var name: String { parameter.name }
4949

5050
/// The name of the parameter sanitized to be a valid Swift identifier.
51-
var variableName: String { context.asSwiftSafeName(name) }
51+
var variableName: String { context.safeNameGenerator.swiftMemberName(for: name) }
5252

5353
/// A Boolean value that indicates whether the parameter must be specified
5454
/// when performing the OpenAPI operation.

Sources/_OpenAPIGeneratorCore/Translator/RequestBody/translateRequestBody.swift

+3-3
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ extension TypesFileTranslator {
5252
bodyMembers.append(contentsOf: inlineTypeDecls)
5353
}
5454
let contentType = content.content.contentType
55-
let identifier = typeAssigner.contentSwiftName(contentType)
55+
let identifier = context.safeNameGenerator.swiftContentTypeName(for: contentType)
5656
let associatedType = content.resolvedTypeUsage.withOptional(false)
5757
let contentCase: Declaration = .commentable(
5858
contentType.docComment(typeName: contentTypeName),
@@ -148,7 +148,7 @@ extension ClientFileTranslator {
148148
var cases: [SwitchCaseDescription] = try contents.map { typedContent in
149149
let content = typedContent.content
150150
let contentType = content.contentType
151-
let contentTypeIdentifier = typeAssigner.contentSwiftName(contentType)
151+
let contentTypeIdentifier = context.safeNameGenerator.swiftContentTypeName(for: contentType)
152152
let contentTypeHeaderValue = contentType.headerValueForSending
153153

154154
let extraBodyAssignArgs: [FunctionArgumentDescription]
@@ -251,7 +251,7 @@ extension ServerFileTranslator {
251251
argumentNames: ["value"],
252252
body: [
253253
.expression(
254-
.dot(typeAssigner.contentSwiftName(typedContent.content.contentType))
254+
.dot(context.safeNameGenerator.swiftContentTypeName(for: typedContent.content.contentType))
255255
.call([.init(label: nil, expression: .identifierPattern("value"))])
256256
)
257257
]

Sources/_OpenAPIGeneratorCore/Translator/Responses/TypedResponseHeader.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ struct TypedResponseHeader {
3939
extension TypedResponseHeader {
4040

4141
/// The name of the header sanitized to be a valid Swift identifier.
42-
var variableName: String { context.asSwiftSafeName(name) }
42+
var variableName: String { context.safeNameGenerator.swiftMemberName(for: name) }
4343

4444
/// A Boolean value that indicates whether the response header can
4545
/// be omitted in the HTTP response.

Sources/_OpenAPIGeneratorCore/Translator/Responses/translateResponse.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,7 @@ extension TypesFileTranslator {
141141
) throws -> [Declaration] {
142142
var bodyCases: [Declaration] = []
143143
let contentType = typedContent.content.contentType
144-
let identifier = typeAssigner.contentSwiftName(contentType)
144+
let identifier = context.safeNameGenerator.swiftContentTypeName(for: contentType)
145145
let associatedType = typedContent.resolvedTypeUsage
146146
let content = typedContent.content
147147
let schema = content.schema

0 commit comments

Comments
 (0)