diff --git a/FrontEnd/styles/images.scss b/FrontEnd/styles/images.scss index 4d1662990..de3954111 100644 --- a/FrontEnd/styles/images.scss +++ b/FrontEnd/styles/images.scss @@ -34,6 +34,7 @@ --image-download: url(''); --image-error: url(''); --image-executables: url(''); + --image-fork: url(''); --image-ghcta-header: url(''); --image-github: url(''); --image-heart: url(''); @@ -73,6 +74,7 @@ --image-download: url(''); --image-error: url(''); --image-executables: url(''); + --image-fork: url(''); --image-ghcta-header: url(''); --image-github: url(''); --image-heart: url(''); diff --git a/FrontEnd/styles/package.scss b/FrontEnd/styles/package.scss index b32d43b54..28ee28c6b 100644 --- a/FrontEnd/styles/package.scss +++ b/FrontEnd/styles/package.scss @@ -75,6 +75,11 @@ background-image: var(--image-warning); } + li.forked { + grid-column-start: span 2; + background-image: var(--image-fork); + } + li.authors { grid-column-start: span 2; background-image: var(--image-authors); diff --git a/Resources/SVGs/fork~dark.svg b/Resources/SVGs/fork~dark.svg new file mode 100644 index 000000000..c4ba52a15 --- /dev/null +++ b/Resources/SVGs/fork~dark.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/Resources/SVGs/fork~light.svg b/Resources/SVGs/fork~light.svg new file mode 100644 index 000000000..534c94039 --- /dev/null +++ b/Resources/SVGs/fork~light.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/Sources/App/Controllers/API/API+PackageController+GetRoute+Model.swift b/Sources/App/Controllers/API/API+PackageController+GetRoute+Model.swift index 211e5fb17..f24c59228 100644 --- a/Sources/App/Controllers/API/API+PackageController+GetRoute+Model.swift +++ b/Sources/App/Controllers/API/API+PackageController+GetRoute+Model.swift @@ -50,6 +50,7 @@ extension API.PackageController.GetRoute { var releaseReferences: [App.Version.Kind: App.Reference] var fundingLinks: [FundingLink] var swift6Readiness: Swift6Readiness? + var forkedFromInfo: ForkedFromInfo? internal init(packageId: Package.Id, repositoryOwner: String, @@ -81,7 +82,8 @@ extension API.PackageController.GetRoute { releaseReference: App.Reference?, preReleaseReference: App.Reference?, fundingLinks: [FundingLink] = [], - swift6Readiness: Swift6Readiness? + swift6Readiness: Swift6Readiness?, + forkedFromInfo: ForkedFromInfo? ) { self.packageId = packageId self.repositoryOwner = repositoryOwner @@ -123,6 +125,7 @@ extension API.PackageController.GetRoute { }() self.fundingLinks = fundingLinks self.swift6Readiness = swift6Readiness + self.forkedFromInfo = forkedFromInfo } init?(result: API.PackageController.PackageResult, @@ -132,7 +135,8 @@ extension API.PackageController.GetRoute { swiftVersionBuildInfo: BuildInfo?, platformBuildInfo: BuildInfo?, weightedKeywords: [WeightedKeyword] = [], - swift6Readiness: Swift6Readiness?) { + swift6Readiness: Swift6Readiness?, + forkedFromInfo: ForkedFromInfo?) { // we consider certain attributes as essential and return nil (raising .notFound) let repository = result.repository guard @@ -177,7 +181,8 @@ extension API.PackageController.GetRoute { releaseReference: result.releaseVersion?.reference, preReleaseReference: result.preReleaseVersion?.reference, fundingLinks: result.repository.fundingLinks, - swift6Readiness: swift6Readiness + swift6Readiness: swift6Readiness, + forkedFromInfo: forkedFromInfo ) } @@ -348,7 +353,14 @@ extension API.PackageController.GetRoute.Model { } } } - + + enum ForkedFromInfo: Codable, Equatable { + case fromSPI(originalOwner: String, + originalOwnerName: String, + originalRepo: String, + originalPackageName: String) + case fromGitHub(url: String) + } } diff --git a/Sources/App/Controllers/API/API+PackageController+GetRoute.swift b/Sources/App/Controllers/API/API+PackageController+GetRoute.swift index 1a7d6750a..782967b6c 100644 --- a/Sources/App/Controllers/API/API+PackageController+GetRoute.swift +++ b/Sources/App/Controllers/API/API+PackageController+GetRoute.swift @@ -45,6 +45,7 @@ extension API.PackageController { async let buildInfo = API.PackageController.BuildInfo.query(on: database, owner: owner, repository: repository) + async let forkedFromInfo = forkedFromInfo(on: database, fork: packageResult.repository.forkedFrom) guard let model = try await Self.Model( @@ -55,7 +56,8 @@ extension API.PackageController { swiftVersionBuildInfo: buildInfo.swiftVersion, platformBuildInfo: buildInfo.platform, weightedKeywords: weightedKeywords, - swift6Readiness: buildInfo.swift6Readiness + swift6Readiness: buildInfo.swift6Readiness, + forkedFromInfo: forkedFromInfo ), let schema = API.PackageSchema(result: packageResult) else { @@ -84,4 +86,34 @@ extension API.PackageController.GetRoute { beta: links[1], latest: links[2]) } + + static func forkedFromInfo(on database: Database, fork: Fork?) async -> Model.ForkedFromInfo? { + guard let forkedFrom = fork else { return nil } + switch forkedFrom { + case .parentId(let id, let fallbackURL): + return await Model.ForkedFromInfo.query(on: database, packageId: id, fallbackURL: fallbackURL) + case let .parentURL(url): + return .fromGitHub(url: url) + } + } +} + + +extension API.PackageController.GetRoute.Model.ForkedFromInfo { + static func query(on database: Database, packageId: Package.Id, fallbackURL: String) async -> Self? { + let model = try? await Joined3 + .query(on: database, packageId: packageId, version: .defaultBranch) + .first() + + guard let repoName = model?.repository.name, + let ownerName = model?.repository.ownerName, + let owner = model?.repository.owner else { + return .fromGitHub(url: fallbackURL) + } + + return .fromSPI(originalOwner: owner, + originalOwnerName: ownerName, + originalRepo: repoName, + originalPackageName: model?.version.packageName ?? repoName) + } } diff --git a/Sources/App/Controllers/API/Types+WithExample.swift b/Sources/App/Controllers/API/Types+WithExample.swift index 1741e5dec..a15170f81 100644 --- a/Sources/App/Controllers/API/Types+WithExample.swift +++ b/Sources/App/Controllers/API/Types+WithExample.swift @@ -247,7 +247,8 @@ extension API.PackageController.GetRoute.Model: WithExample { defaultBranchReference: .branch("main"), releaseReference: .tag(1, 2, 3, "1.2.3"), preReleaseReference: nil, - swift6Readiness: nil) + swift6Readiness: nil, + forkedFromInfo: nil) } } diff --git a/Sources/App/Core/Query+Support/Joined3+Package.swift b/Sources/App/Core/Query+Support/Joined3+Package.swift index 1b9d11a36..9c67e3bac 100644 --- a/Sources/App/Core/Query+Support/Joined3+Package.swift +++ b/Sources/App/Core/Query+Support/Joined3+Package.swift @@ -40,4 +40,9 @@ extension Joined3 where M == Package, R1 == Repository, R2 == Version { .filter(Repository.self, \.$owner, .custom("ilike"), owner) .filter(Repository.self, \.$name, .custom("ilike"), repository) } + + static func query(on database: Database, packageId: Package.Id, version: Version.Kind) -> JoinedQueryBuilder { + query(on: database, version: version) + .filter(Package.self, \Package.$id == packageId) + } } diff --git a/Sources/App/Models/Repository.swift b/Sources/App/Models/Repository.swift index 22338a08d..1b252f83b 100644 --- a/Sources/App/Models/Repository.swift +++ b/Sources/App/Models/Repository.swift @@ -224,6 +224,7 @@ final class Repository: @unchecked Sendable, Model, Content { .filter(\.$package.$id == pkgId) .first() ?? Repository(packageId: pkgId) } + } diff --git a/Sources/App/Views/PackageController/GetRoute.Model+ext.swift b/Sources/App/Views/PackageController/GetRoute.Model+ext.swift index d4e77d1ef..b647d1696 100644 --- a/Sources/App/Views/PackageController/GetRoute.Model+ext.swift +++ b/Sources/App/Views/PackageController/GetRoute.Model+ext.swift @@ -181,6 +181,50 @@ extension API.PackageController.GetRoute.Model { return .empty } } + + func forkedListItem() -> Node { + if let forkedFromInfo { + let item: Node = { + switch forkedFromInfo { + case .fromGitHub(let url): + var text = url.replacingOccurrences(of: "https://github.com/", with: "") + text = text.removingSuffix(".git") + let repoURLNode: Node = .a( + .href(url), + .text(text) + ) + return .group( + .text("Forked from "), + repoURLNode, + .text(".") + ) + case .fromSPI(_, let ownerName, _, let originalPackageName): + let repoURLNode: Node = .a( + .href(forkedFromInfo.url), + .text("\(originalPackageName)") + ) + let ownerNode: Node = .a( + .href(forkedFromInfo.ownerURL ?? ""), + .text("\(ownerName)") + ) + return .group( + .text("Forked from "), + repoURLNode, + .text(" by "), + ownerNode, + .text(".") + ) + } + }() + + return .li( + .class("forked"), + item + ) + } else { + return .empty + } + } func binaryTargetsItem() -> Node { guard hasBinaryTargets else { return .empty } @@ -667,3 +711,24 @@ extension API.PackageController.GetRoute.Model.Swift6Readiness { return lines.joined(separator: "\n") } } + + +extension API.PackageController.GetRoute.Model.ForkedFromInfo { + var url: String { + switch self { + case .fromSPI(let originalOwner, _, let originalRepo, _): + return SiteURL.package(.value(originalOwner), .value(originalRepo), nil).relativeURL() + case .fromGitHub(let url): + return url + } + } + + var ownerURL: String? { + switch self { + case .fromSPI(let owner, _, _, _): + return SiteURL.author(.value(owner)).relativeURL() + case .fromGitHub: + return nil + } + } +} diff --git a/Sources/App/Views/PackageController/PackageShow+View.swift b/Sources/App/Views/PackageController/PackageShow+View.swift index 37f4a4d84..271e68543 100644 --- a/Sources/App/Views/PackageController/PackageShow+View.swift +++ b/Sources/App/Views/PackageController/PackageShow+View.swift @@ -167,6 +167,7 @@ extension PackageShow { .ul( .class("main-metadata"), model.archivedListItem(), + model.forkedListItem(), model.authorsListItem(), model.binaryTargetsItem(), model.historyListItem(), diff --git a/Tests/AppTests/API+PackageController+GetRoute+ModelTests.swift b/Tests/AppTests/API+PackageController+GetRoute+ModelTests.swift index bff7618a4..f5d242a31 100644 --- a/Tests/AppTests/API+PackageController+GetRoute+ModelTests.swift +++ b/Tests/AppTests/API+PackageController+GetRoute+ModelTests.swift @@ -42,7 +42,8 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { swiftVersionBuildInfo: nil, platformBuildInfo: nil, weightedKeywords: [], - swift6Readiness: nil) + swift6Readiness: nil, + forkedFromInfo: nil) // validate XCTAssertNotNil(m) @@ -64,7 +65,8 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { swiftVersionBuildInfo: nil, platformBuildInfo: nil, weightedKeywords: [], - swift6Readiness: nil)) + swift6Readiness: nil, + forkedFromInfo: nil)) // validate XCTAssertEqual(model.packageIdentity, "swift-bar") @@ -86,7 +88,8 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { swiftVersionBuildInfo: nil, platformBuildInfo: nil, weightedKeywords: [], - swift6Readiness: nil)) + swift6Readiness: nil, + forkedFromInfo: nil)) // validate XCTAssertEqual(model.documentationTarget, .internal(docVersion: .reference("main"), archive: "archive1")) @@ -112,11 +115,40 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { swiftVersionBuildInfo: nil, platformBuildInfo: nil, weightedKeywords: [], - swift6Readiness: nil)) + swift6Readiness: nil, + forkedFromInfo: nil)) // validate XCTAssertEqual(model.documentationTarget, .external(url: "https://example.com/package/documentation")) } + + func test_ForkedFromInfo_query() async throws { + let originalPkg = try await savePackage(on: app.db, id: .id0, "https://github.com/original/original") + try await Repository(package: originalPkg, + name: "original", + owner: "original", + ownerName: "OriginalOwner").save(on: app.db) + try await App.Version(package: originalPkg, latest: .defaultBranch, packageName: "OriginalPkg", reference: .branch("main")) + .save(on: app.db) + + // MUT + let forkedFrom = await API.PackageController.GetRoute.Model.ForkedFromInfo.query(on: app.db, packageId: .id0, fallbackURL: "https://github.com/original/original.git") + + // validate + XCTAssertEqual(forkedFrom, .fromSPI(originalOwner: "original", + originalOwnerName: "OriginalOwner", + originalRepo: "original", + originalPackageName: "OriginalPkg")) + } + + func test_ForkedFromInfo_query_fallback() async throws { + // when the package can't be found resort to fallback URL + // MUT + let forkedFrom = await API.PackageController.GetRoute.Model.ForkedFromInfo.query(on: app.db, packageId: .id0, fallbackURL: "https://github.com/original/original.git") + + // validate + XCTAssertEqual(forkedFrom, .fromGitHub(url: "https://github.com/original/original.git")) + } func test_gitHubOwnerUrl() throws { var model = API.PackageController.GetRoute.Model.mock @@ -144,6 +176,37 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { let renderedHistory = model.historyListItem().render(indentedBy: .spaces(2)) assertSnapshot(of: renderedHistory, as: .lines) } + + func test_forked_from_github() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromGitHub(url: "https://github.com/owner/repository.git") + let renderedForkedFrom = model.forkedListItem().render(indentedBy: .spaces(2)) + assertSnapshot(of: renderedForkedFrom, as: .lines) + } + + func test_forked_from_spi_same_package_name() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromSPI( + originalOwner: "owner", + originalOwnerName: "OriginalOwner", + originalRepo: "repo", + originalPackageName: "Test" + ) + let renderedForkedFrom = model.forkedListItem().render(indentedBy: .spaces(2)) + assertSnapshot(of: renderedForkedFrom, as: .lines) + } + + func test_forked_from_spi_different_package_name() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromSPI( + originalOwner: "owner", + originalOwnerName: "OriginalOwner", + originalRepo: "repo", + originalPackageName: "Different" + ) + let renderedForkedFrom = model.forkedListItem().render(indentedBy: .spaces(2)) + assertSnapshot(of: renderedForkedFrom, as: .lines) + } func test_binary_targets() throws { var model = API.PackageController.GetRoute.Model.mock @@ -329,6 +392,41 @@ class API_PackageController_GetRoute_ModelTests: SnapshotTestCase { .fromSPIManifest("By Author One, Author Two, and more!") XCTAssertEqual(model.authorsListItem().render(), "
  • By Author One, Author Two, and more!
  • ") } + + func test_forkedFrom_github_formatting() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromGitHub(url: "https://github.com/owner/repository.git") + let renderedForkedFrom = model.forkedListItem().render() + XCTAssertEqual(renderedForkedFrom, "
  • Forked from owner/repository.
  • ") + } + + func test_forkedFrom_spi_same_package_name_formatting() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromSPI( + originalOwner: "owner", + originalOwnerName: "OriginalOwner", + originalRepo: "repo", + originalPackageName: "Test" + ) + let url = SiteURL.package(.value("owner"), .value("repo"), nil).relativeURL() + let ownerUrl = model.forkedFromInfo?.ownerURL ?? "" + let renderedForkedFrom = model.forkedListItem().render() + XCTAssertEqual(renderedForkedFrom, "
  • Forked from Test by OriginalOwner.
  • ") + } + + func test_forkedFrom_spi_different_package_name_formatting() throws { + var model = API.PackageController.GetRoute.Model.mock + model.forkedFromInfo = .fromSPI( + originalOwner: "owner", + originalOwnerName: "OriginalOwner", + originalRepo: "repo", + originalPackageName: "Different" + ) + let url = SiteURL.package(.value("owner"), .value("repo"), nil).relativeURL() + let ownerUrl = model.forkedFromInfo?.ownerURL ?? "" + let renderedForkedFrom = model.forkedListItem().render() + XCTAssertEqual(renderedForkedFrom, "
  • Forked from Different by OriginalOwner.
  • ") + } func test_BuildInfo_init() throws { // ensure nil propagation when all versions' values are nil diff --git a/Tests/AppTests/Mocks/API.PackageController.GetRoute.Model+mock.swift b/Tests/AppTests/Mocks/API.PackageController.GetRoute.Model+mock.swift index cafa1c83d..31da3f38e 100644 --- a/Tests/AppTests/Mocks/API.PackageController.GetRoute.Model+mock.swift +++ b/Tests/AppTests/Mocks/API.PackageController.GetRoute.Model+mock.swift @@ -125,7 +125,8 @@ extension API.PackageController.GetRoute.Model { defaultBranchReference: .branch("main"), releaseReference: .tag(5, 2, 0), preReleaseReference: .tag(5, 3, 0, "beta.1"), - swift6Readiness: nil + swift6Readiness: nil, + forkedFromInfo: nil ) } } diff --git a/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from.1.txt b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from.1.txt new file mode 100644 index 000000000..620ce3bbe --- /dev/null +++ b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from.1.txt @@ -0,0 +1,3 @@ +
  • Forked from + repository. +
  • \ No newline at end of file diff --git a/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_github.1.txt b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_github.1.txt new file mode 100644 index 000000000..be633d8e1 --- /dev/null +++ b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_github.1.txt @@ -0,0 +1,3 @@ +
  • Forked from + owner/repository. +
  • \ No newline at end of file diff --git a/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_different_package_name.1.txt b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_different_package_name.1.txt new file mode 100644 index 000000000..c97d0da5a --- /dev/null +++ b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_different_package_name.1.txt @@ -0,0 +1,4 @@ +
  • Forked from + Different by + OriginalOwner. +
  • \ No newline at end of file diff --git a/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_same_package_name.1.txt b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_same_package_name.1.txt new file mode 100644 index 000000000..eb55eff80 --- /dev/null +++ b/Tests/AppTests/__Snapshots__/API+PackageController+GetRoute+ModelTests/test_forked_from_spi_same_package_name.1.txt @@ -0,0 +1,4 @@ +
  • Forked from + Test by + OriginalOwner. +
  • \ No newline at end of file