major refactor to repo

This commit is contained in:
Ryan Nystrom
2017-09-03 17:27:26 -04:00
parent 3ffc4f2b9c
commit e476976fd3
17 changed files with 792 additions and 404 deletions

View File

@@ -8,139 +8,195 @@
import Apollo
protocol RepositoryLoadable {
var owner: String { get }
var name: String { get }
var hasIssuesEnabled: Bool { get }
}
protocol RepositoryQuery {
init(owner: String, name: String, after: String?)
// generated queries should share the same init
func summaryTypes(from data: GraphQLMappable) -> [IssueSummaryType]
func nextPageToken(from data: GraphQLMappable) -> String?
}
extension RepoIssuesQuery: RepositoryQuery {
extension RepoIssuePagesQuery: RepositoryQuery {
func summaryTypes(from data: GraphQLMappable) -> [IssueSummaryType] {
guard let issues = data as? RepoIssuesQuery.Data else { return [] }
guard let issues = data as? Data else { return [] }
return issues.repository?.issues.nodes?.flatMap { $0 } ?? []
}
func nextPageToken(from data: GraphQLMappable) -> String? {
guard let issues = data as? RepoIssuesQuery.Data else { return nil }
guard let issues = data as? Data else { return nil }
guard let pageInfo = issues.repository?.issues.pageInfo, pageInfo.hasNextPage else { return nil }
return pageInfo.endCursor
}
}
extension RepoPullRequestsQuery: RepositoryQuery {
extension RepoPullRequestPagesQuery: RepositoryQuery {
func summaryTypes(from data: GraphQLMappable) -> [IssueSummaryType] {
guard let prs = data as? RepoPullRequestsQuery.Data else { return [] }
guard let prs = data as? RepoPullRequestPagesQuery.Data else { return [] }
return prs.repository?.pullRequests.nodes?.flatMap { $0 } ?? []
}
func nextPageToken(from data: GraphQLMappable) -> String? {
guard let prs = data as? RepoPullRequestsQuery.Data else { return nil }
guard let prs = data as? RepoPullRequestPagesQuery.Data else { return nil }
guard let pageInfo = prs.repository?.pullRequests.pageInfo, pageInfo.hasNextPage else { return nil }
return pageInfo.endCursor
}
}
func createSummaryModel(_ node: IssueSummaryType, containerWidth: CGFloat) -> IssueSummaryModel? {
guard let date = GithubAPIDateFormatter().date(from: node.repoEventFields.createdAt)
else { return nil }
let attributes = [
NSFontAttributeName: Styles.Fonts.title,
NSForegroundColorAttributeName: Styles.Colors.Gray.dark.color
]
let title = NSAttributedStringSizing(
containerWidth: containerWidth,
attributedText: NSAttributedString(string: node.title, attributes: attributes),
inset: RepositorySummaryCell.titleInset
)
return IssueSummaryModel(
id: node.id,
title: title,
number: node.number,
created: date,
author: node.repoEventFields.author?.login ?? Strings.unknown,
status: node.status,
pullRequest: node.pullRequest
)
}
func createSummaryModel(
query: RepositoryQuery,
data: GraphQLMappable,
containerWidth: CGFloat
) -> (models: [IssueSummaryModel], nextPage: String?) {
let nextPage = query.nextPageToken(from: data)
let models: [IssueSummaryModel] = query.summaryTypes(from: data).flatMap { (node: IssueSummaryType) in
return createSummaryModel(node, containerWidth: containerWidth)
}
return (models, nextPage)
}
final class RepositoryClient {
var issues = [IssueSummaryModel]()
var issuesNextPage: String?
var pullRequests = [IssueSummaryModel]()
var pullRequestsNextPage: String?
let githubClient: GithubClient
let repo: RepositoryLoadable
init(githubClient: GithubClient, repo: RepositoryLoadable) {
private let owner: String
private let name: String
init(githubClient: GithubClient, owner: String, name: String) {
self.githubClient = githubClient
self.repo = repo
self.owner = owner
self.name = name
}
struct RepositoryLoadSuccessPayload {
let models: [IssueSummaryModel]?
struct RepositoryPayload {
let models: [IssueSummaryModel]
let nextPage: String?
}
enum RepoLoadResultType {
case error
case success(RepositoryLoadSuccessPayload)
}
private func load<T: GraphQLQuery>(queryType: T.Type,
repo: RepositoryLoadable,
after: String? = nil,
containerWidth: CGFloat,
completion: @escaping (RepoLoadResultType) -> ()) where T: RepositoryQuery {
let query = queryType.init(owner: repo.owner, name: repo.name, after: after)
private func loadPage<T: GraphQLQuery>(
query: T,
containerWidth: CGFloat,
completion: @escaping (Result<RepositoryPayload>) -> ()
) where T: RepositoryQuery {
githubClient.apollo.fetch(query: query, cachePolicy: .fetchIgnoringCacheData) { (result, error) in
guard error == nil, result?.errors == nil, let data = result?.data else {
ShowErrorStatusBar(graphQLErrors: result?.errors, networkError: error)
completion(.error)
completion(.error(nil))
return
}
let summaries: [IssueSummaryModel] = query.summaryTypes(from: data).map { summaryType in
summaryType.attributedTitle.computeSize(containerWidth)
return IssueSummaryModel(info: summaryType)
}
completion(.success(RepositoryLoadSuccessPayload(models: summaries, nextPage: query.nextPageToken(from: data))))
}
}
func loadMoreIssues(containerWidth: CGFloat, completion: @escaping () -> ()) {
guard repo.hasIssuesEnabled else { return }
load(queryType: RepoIssuesQuery.self, repo: repo, after: issuesNextPage, containerWidth: containerWidth) { result in
switch result {
case .error:
completion()
case .success(let payload):
if let models = payload.models, models.count > 0 {
self.issues += models
DispatchQueue.global().async {
// jump to a bg queue to parse models and presize text
let summary = createSummaryModel(
query: query,
data: data,
containerWidth: containerWidth
)
DispatchQueue.main.async {
completion(.success(RepositoryPayload(
models: summary.models,
nextPage: summary.nextPage
)))
}
self.issuesNextPage = payload.nextPage
completion()
}
}
}
func loadMorePullRequests(containerWidth: CGFloat, completion: @escaping () -> ()) {
load(queryType: RepoPullRequestsQuery.self, repo: repo, after: pullRequestsNextPage, containerWidth: containerWidth) { result in
switch result {
case .error:
completion()
case .success(let payload):
if let models = payload.models, models.count > 0 {
self.pullRequests += models
}
self.pullRequestsNextPage = payload.nextPage
completion()
}
}
func loadMoreIssues(
nextPage: String?,
containerWidth: CGFloat,
completion: @escaping (Result<RepositoryPayload>) -> ()
) {
loadPage(
query: RepoIssuePagesQuery(owner: owner, name: name, after: nextPage, pageSize: 100),
containerWidth: containerWidth,
completion: completion
)
}
func load(containerWidth: CGFloat, completion: @escaping () -> ()) {
var expectedResponseCount = repo.hasIssuesEnabled ? 2 : 1
let checkResponses = {
expectedResponseCount -= 1
guard expectedResponseCount == 0 else { return }
completion()
func loadMorePullRequests(
nextPage: String?,
containerWidth: CGFloat,
completion: @escaping (Result<RepositoryPayload>) -> ()
) {
loadPage(
query: RepoPullRequestPagesQuery(owner: owner, name: name, after: nextPage, pageSize: 100),
containerWidth: containerWidth,
completion: completion
)
}
struct RepositoryDetailsPayload {
let issues: RepositoryPayload
let pullRequests: RepositoryPayload
}
func load(
containerWidth: CGFloat,
completion: @escaping (Result<RepositoryDetailsPayload>) -> ()
) {
let query = RepoDetailsQuery(owner: owner, name: name, pageSize: 100)
githubClient.apollo.fetch(query: query, cachePolicy: .fetchIgnoringCacheData) { (result, error) in
guard error == nil, result?.errors == nil, let data = result?.data else {
ShowErrorStatusBar(graphQLErrors: result?.errors, networkError: error)
completion(.error(nil))
return
}
let issueNodes = (data.repository?.issues.nodes ?? []).flatMap { $0 }
let issues = issueNodes.flatMap { (node: IssueSummaryType) in
return createSummaryModel(node, containerWidth: containerWidth)
}
let issueNextPage: String?
if let pageInfo = data.repository?.issues.pageInfo, pageInfo.hasNextPage == true {
issueNextPage = pageInfo.endCursor
} else {
issueNextPage = nil
}
let prNodes = (data.repository?.pullRequests.nodes ?? []).flatMap { $0 }
let prs = prNodes.flatMap { (node: IssueSummaryType) in
return createSummaryModel(node, containerWidth: containerWidth)
}
let prNextPage: String?
if let pageInfo = data.repository?.issues.pageInfo, pageInfo.hasNextPage == true {
prNextPage = pageInfo.endCursor
} else {
prNextPage = nil
}
completion(.success(RepositoryDetailsPayload(
issues: RepositoryPayload(models: issues, nextPage: issueNextPage),
pullRequests: RepositoryPayload(models: prs, nextPage: prNextPage)
)))
}
loadMoreIssues(containerWidth: containerWidth, completion: checkResponses)
loadMorePullRequests(containerWidth: containerWidth, completion: checkResponses)
}
}