Skip to content
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

Update to Swift 6 #109

Merged
merged 45 commits into from
Dec 13, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
acf889c
Update to Swift 5.10
fpseverino Nov 10, 2024
0b90625
Update ImperialCore
fpseverino Nov 10, 2024
512cb80
Update Services
fpseverino Nov 10, 2024
7b86214
Remove old Microsoft service
fpseverino Nov 10, 2024
954b706
Update to Swift 6
fpseverino Nov 10, 2024
7efd4cb
Revert to macOS 13
fpseverino Nov 10, 2024
13bcee8
Update Keycloak.swift
fpseverino Nov 14, 2024
f3d5d1f
Update Dropbox.swift
fpseverino Nov 14, 2024
7126c1f
Update Discord.swift
fpseverino Nov 14, 2024
1daa432
Update FederatedService.swift
fpseverino Nov 14, 2024
406a65d
Update FederatedServiceRouter.swift
fpseverino Nov 14, 2024
967b3e5
Update FederatedServiceRouter.swift
fpseverino Nov 14, 2024
9933bc0
Update Auth0.swift
fpseverino Nov 14, 2024
7a85b7d
Conform all services
fpseverino Nov 14, 2024
4812c42
Remove `Optional+Imperial`
fpseverino Nov 14, 2024
2bb8422
Change existential return types to opaque (where possible)
fpseverino Nov 14, 2024
91abbb3
Remove `FullTypedThrows`
fpseverino Nov 14, 2024
45634e6
Remove superfluous `String` extension
fpseverino Nov 15, 2024
88553a3
Improve error types
fpseverino Nov 15, 2024
c824ad4
Introduce basic testing
fpseverino Nov 16, 2024
f3e2e60
Improve `OAuthService` sendability
fpseverino Nov 16, 2024
c9199fc
Add more basic tests
fpseverino Nov 16, 2024
9732bbf
Make a lot of stuff `Sendable`
fpseverino Nov 17, 2024
a11f7c8
Remove a lot of `Sendable` warnings
fpseverino Nov 17, 2024
d70b8ea
Make `ShopifyRouter` fully `Sendable`
fpseverino Nov 20, 2024
b368cee
Enable Swift 6 language mode
fpseverino Nov 20, 2024
f1fef48
Update to JWTKit v5
fpseverino Nov 20, 2024
c17faff
Add formatting and linting
fpseverino Nov 20, 2024
0d08b17
Update README and add `.spi.yml`
fpseverino Nov 20, 2024
7a59e31
Adjust formatting and update documentation
fpseverino Nov 22, 2024
cb9f630
Make error testing more type safe
fpseverino Nov 22, 2024
0ab3d8a
Update Sources/ImperialShopify/ShopifyRouter.swift
fpseverino Nov 22, 2024
cc43d63
Make all `FederatedService` `Sendable`
fpseverino Nov 22, 2024
87c761b
Refactor `Shopify` class to streamline token and router properties
fpseverino Nov 22, 2024
7d1550f
Remove `baseURL` where unnecessary
fpseverino Nov 22, 2024
27611f7
Add OAuth routes inside `withApp`
fpseverino Nov 22, 2024
cb6e3e5
Improve tests
fpseverino Nov 23, 2024
c6163c3
Make the linter happy
fpseverino Nov 23, 2024
2e9ef29
Update remaining services
fpseverino Dec 4, 2024
6bee2ff
Remove unnecessary `public`s
fpseverino Dec 4, 2024
83b6160
Update .spi.yml
fpseverino Dec 4, 2024
169c111
Make the linter happy
fpseverino Dec 4, 2024
e36ae31
Remove `FederatedCreatable` and `OAuthService`
fpseverino Dec 4, 2024
70f39d0
Change all `class`es to `struct`s
fpseverino Dec 4, 2024
4b3e1ba
Make a lot of `public` APIs `internal` or `package`
fpseverino Dec 6, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
56 changes: 10 additions & 46 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
@@ -1,49 +1,13 @@
name: test
concurrency:
group: ${{ github.workflow }}-${{ github.ref }}
cancel-in-progress: true
on:
push:
branches: [ main ]
pull_request:
branches: [ main ]

pull_request: { types: [opened, reopened, synchronize, ready_for_review] }
push: { branches: [ main ] }

jobs:
macos:
strategy:
fail-fast: false
matrix:
xcode: [latest, latest-stable]
runs-on: macos-latest
steps:
- name: Select latest available Xcode
uses: maxim-lobanov/[email protected]
with: { 'xcode-version': '${{ matrix.xcode }}' }
- name: Check out code
uses: actions/checkout@v2
- name: Run tests with Thread Sanitizer
run: swift test --enable-test-discovery --sanitize=thread
linux:
strategy:
fail-fast: false
matrix:
swiftver:
- swift:5.2
- swift:5.3
swiftos:
- xenial
- bionic
- focal
- centos7
- centos8
- amazonlinux2
container: ${{ format('{0}-{1}', matrix.swiftver, matrix.swiftos) }}
runs-on: ubuntu-latest
steps:
- name: SPM is incompatible with CentOS 7
if: ${{ matrix.swiftos == 'centos7' }}
run: |
yum install -y make libcurl-devel
git clone https://github.com/git/git -bv2.28.0 --depth 1 && cd git
make prefix=/usr -j all install NO_OPENSSL=1 NO_EXPAT=1 NO_TCLTK=1 NO_GETTEXT=1 NO_PERL=1
- name: Check out code
uses: actions/checkout@v2
- name: Run tests with Thread Sanitizer
run: swift test --enable-test-discovery --sanitize=thread
unit-tests:
uses: vapor/ci/.github/workflows/run-unit-tests.yml@main
secrets:
CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
39 changes: 24 additions & 15 deletions Package.swift
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
// swift-tools-version:5.2
// swift-tools-version:6.0
import PackageDescription

let package = Package(
name: "Imperial",
platforms: [
.macOS(.v10_15)
.macOS(.v14)
],
fpseverino marked this conversation as resolved.
Show resolved Hide resolved
products: [
.library(name: "ImperialCore", targets: ["ImperialCore"]),
Expand Down Expand Up @@ -42,18 +42,27 @@ let package = Package(
dependencies: [
.product(name: "Vapor", package: "vapor"),
.product(name: "JWTKit", package: "jwt-kit"),
]
fpseverino marked this conversation as resolved.
Show resolved Hide resolved
],
swiftSettings: swiftSettings
),
.target(name: "ImperialAuth0", dependencies: ["ImperialCore"]),
.target(name: "ImperialDiscord", dependencies: ["ImperialCore"]),
.target(name: "ImperialDropbox", dependencies: ["ImperialCore"]),
.target(name: "ImperialFacebook", dependencies: ["ImperialCore"]),
.target(name: "ImperialGitHub", dependencies: ["ImperialCore"]),
.target(name: "ImperialGitlab", dependencies: ["ImperialCore"]),
.target(name: "ImperialGoogle", dependencies: ["ImperialCore"]),
.target(name: "ImperialKeycloak", dependencies: ["ImperialCore"]),
.target(name: "ImperialMicrosoft", dependencies: ["ImperialCore"]),
.target(name: "ImperialShopify", dependencies: ["ImperialCore"]),
.testTarget(name: "ImperialTests", dependencies: ["ImperialCore", "ImperialShopify"]),
]
.target(name: "ImperialAuth0", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialDiscord", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialDropbox", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialFacebook", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialGitHub", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialGitlab", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialGoogle", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialKeycloak", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialMicrosoft", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.target(name: "ImperialShopify", dependencies: ["ImperialCore"], swiftSettings: swiftSettings),
.testTarget(name: "ImperialTests", dependencies: ["ImperialCore", "ImperialShopify"], swiftSettings: swiftSettings),
],
swiftLanguageModes: [.v5]
)

var swiftSettings: [SwiftSetting] {
[
.enableUpcomingFeature("ExistentialAny"),
.enableUpcomingFeature("FullTypedThrows"),
]
fpseverino marked this conversation as resolved.
Show resolved Hide resolved
}
24 changes: 0 additions & 24 deletions Sources/Imperial/Services/Microsoft/Microsoft.swift
fpseverino marked this conversation as resolved.
Show resolved Hide resolved

This file was deleted.

16 changes: 0 additions & 16 deletions Sources/Imperial/Services/Microsoft/MicrosoftAuth.swift

This file was deleted.

21 changes: 0 additions & 21 deletions Sources/Imperial/Services/Microsoft/MicrosoftCallbackBody.swift

This file was deleted.

81 changes: 0 additions & 81 deletions Sources/Imperial/Services/Microsoft/MicrosoftRouter.swift

This file was deleted.

6 changes: 0 additions & 6 deletions Sources/Imperial/Services/Microsoft/Service+Microsoft.swift

This file was deleted.

10 changes: 5 additions & 5 deletions Sources/ImperialAuth0/Auth0.swift
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,17 @@
import Vapor

public class Auth0: FederatedService {
public var tokens: FederatedServiceTokens
public var router: FederatedServiceRouter
public var tokens: any FederatedServiceTokens
public var router: any FederatedServiceRouter

@discardableResult
public required init(
routes: RoutesBuilder,
routes: any RoutesBuilder,
fpseverino marked this conversation as resolved.
Show resolved Hide resolved
authenticate: String,
authenticateCallback: ((Request) throws -> (EventLoopFuture<Void>))?,
authenticateCallback: ((Request) async throws -> Void)?,
callback: String,
scope: [String] = [],
completion: @escaping (Request, String) throws -> (EventLoopFuture<ResponseEncodable>)
completion: @escaping (Request, String) async throws -> any AsyncResponseEncodable
) throws {
self.router = try Auth0Router(callback: callback, completion: completion)
self.tokens = self.router.tokens
Expand Down
8 changes: 4 additions & 4 deletions Sources/ImperialAuth0/Auth0Router.swift
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,8 @@ import Foundation
public class Auth0Router: FederatedServiceRouter {

public let baseURL: String
public let tokens: FederatedServiceTokens
public let callbackCompletion: (Request, String) throws -> (EventLoopFuture<ResponseEncodable>)
public let tokens: any FederatedServiceTokens
public let callbackCompletion: (Request, String) async throws -> any AsyncResponseEncodable
public var scope: [String] = [ ]
public var requiredScopes = [ "openid" ]
public let callbackURL: String
Expand All @@ -17,7 +17,7 @@ public class Auth0Router: FederatedServiceRouter {
return self.baseURL.finished(with: "/") + path
}

public required init(callback: String, completion: @escaping (Request, String) throws -> (EventLoopFuture<ResponseEncodable>)) throws {
public required init(callback: String, completion: @escaping (Request, String) async throws -> any AsyncResponseEncodable) throws {
let auth = try Auth0Auth()
self.tokens = auth
self.baseURL = "https://\(auth.domain)"
Expand Down Expand Up @@ -45,7 +45,7 @@ public class Auth0Router: FederatedServiceRouter {
return rtn
}

public func callbackBody(with code: String) -> ResponseEncodable {
public func callbackBody(with code: String) -> any AsyncResponseEncodable {
Auth0CallbackBody(clientId: self.tokens.clientID,
clientSecret: self.tokens.clientSecret,
code: code,
Expand Down
2 changes: 1 addition & 1 deletion Sources/ImperialCore/Helpers/Optional+Imperial.swift
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ extension Optional {
/// - Parameter error: The error to throw if the optional is `nil`.
/// - Returns: The value contained in the optional.
/// - Throws: The error passed in if the optional is `nil`.
public func value(or error: Error) throws -> Wrapped {
public func value(or error: any Error) throws -> Wrapped {
fpseverino marked this conversation as resolved.
Show resolved Hide resolved
switch self {
case let .some(value): return value
case .none: throw error
Expand Down
12 changes: 5 additions & 7 deletions Sources/ImperialCore/Helpers/Request+Imperial.swift
Original file line number Diff line number Diff line change
Expand Up @@ -10,18 +10,16 @@ extension Request {
/// - service: The service to get the data from.
/// - Returns: An instance of the type passed in.
/// - Throws: Errors from trying to get the access token from the request.
func create<Model: FederatedCreatable>(_ model: Model.Type, with service: OAuthService, on req: Request) throws -> EventLoopFuture<Model> {
func create<Model: FederatedCreatable>(_ model: Model.Type, with service: OAuthService, on req: Request) async throws -> Model {
let url = try service[model.serviceKey].value(or: ServiceError.noServiceEndpoint(model.serviceKey))

let token = try service.tokenPrefix + req
.accessToken()

return req.client.get(URI(string: url), headers: ["Authorization": token]).flatMap { response in
return try! model.create(from: response).flatMapThrowing { instance in
try self.session.set("imperial-\(model)", to: instance)
return instance
}
}
let response = try await req.client.get(URI(string: url), headers: ["Authorization": token])
let instance = try await model.init(from: response)
try self.session.set("imperial-\(model)", to: instance)
return instance
}

/// Gets an instance of a `FederatedCreatable` type that is stored in the request.
Expand Down
15 changes: 7 additions & 8 deletions Sources/ImperialCore/Middleware/ImperialMiddleware.swift
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import Vapor

/// Protects routes from users without an access token.
public class ImperialMiddleware: Middleware {
public struct ImperialMiddleware: AsyncMiddleware {

/// The path to redirect the user to if they are not authenticated.
let redirectPath: String?
Expand All @@ -15,18 +15,17 @@ public class ImperialMiddleware: Middleware {

/// Checks that the request contains an access token. If it does, let the request through. If not, redirect the user to the `redirectPath`.
/// If the `redirectPath` is `nil`, then throw the error from getting the access token (Abort.unauthorized).
public func respond(to request: Request, chainingTo next: Responder) -> EventLoopFuture<Response> {
public func respond(to request: Request, chainingTo next: any AsyncResponder) async throws -> Response {
do {
_ = try request.accessToken()
return next.respond(to: request)
return try await next.respond(to: request)
} catch let error as Abort where error.status == .unauthorized {
guard let redirectPath = redirectPath else {
return request.eventLoop.makeFailedFuture(error)
guard let redirectPath else {
throw error
}
let redirect: Response = request.redirect(to: redirectPath)
return request.eventLoop.makeSucceededFuture(redirect)
return request.redirect(to: redirectPath)
} catch let error {
return request.eventLoop.makeFailedFuture(error)
throw error
}
}
}
Loading
Loading