// Copyright 2022 The Gitea Authors. All rights reserved. // Use of this source code is governed by a MIT-style // license that can be found in the LICENSE file. package activitypub import ( "context" "errors" "net/url" "strconv" "strings" "code.gitea.io/gitea/models/auth" issues_model "code.gitea.io/gitea/models/issues" repo_model "code.gitea.io/gitea/models/repo" user_model "code.gitea.io/gitea/models/user" "code.gitea.io/gitea/modules/forgefed" repo_module "code.gitea.io/gitea/modules/repository" "code.gitea.io/gitea/modules/setting" "code.gitea.io/gitea/services/activitypub" issue_service "code.gitea.io/gitea/services/issue" pull_service "code.gitea.io/gitea/services/pull" repo_service "code.gitea.io/gitea/services/repository" user_service "code.gitea.io/gitea/services/user" ap "github.com/go-ap/activitypub" ) // Create a new federated user from a Person object func createPerson(ctx context.Context, person *ap.Person) error { name, err := activitypub.PersonIRIToName(person.GetLink()) if err != nil { return err } exists, err := user_model.IsUserExist(ctx, 0, name) if err != nil { return err } if exists { return nil } var email string if person.Location != nil { email = person.Location.GetLink().String() } else { // This might not even work email = strings.ReplaceAll(name, "@", "+") + "@" + setting.Service.NoReplyAddress } if person.PublicKey.PublicKeyPem == "" { return errors.New("person public key not found") } user := &user_model.User{ Name: name, FullName: person.Name.String(), // May not exist!! Email: email, LoginType: auth.Federated, LoginName: person.GetLink().String(), } err = user_model.CreateUser(user) if err != nil { return err } if person.Icon != nil { icon, err := ap.ToObject(person.Icon) if err != nil { return err } iconURL, err := icon.URL.GetLink().URL() if err != nil { return err } body, err := activitypub.Fetch(iconURL) if err != nil { return err } err = user_service.UploadAvatar(user, body) if err != nil { return err } } err = user_model.SetUserSetting(user.ID, user_model.UserActivityPubPrivPem, "") if err != nil { return err } return user_model.SetUserSetting(user.ID, user_model.UserActivityPubPubPem, person.PublicKey.PublicKeyPem) } func createPersonFromIRI(ctx context.Context, personIRI ap.IRI) error { ownerURL, err := url.Parse(personIRI.String()) if err != nil { return err } // Fetch person object resp, err := activitypub.Fetch(ownerURL) if err != nil { return err } // Parse person object ap.ItemTyperFunc = forgefed.GetItemByType ap.JSONItemUnmarshal = forgefed.JSONUnmarshalerFn ap.NotEmptyChecker = forgefed.NotEmpty object, err := ap.UnmarshalJSON(resp) if err != nil { return err } // Create federated user person, err := ap.ToActor(object) if err != nil { return err } return createPerson(ctx, person) } // Create a new federated repo from a Repository object func createRepository(ctx context.Context, repository *forgefed.Repository) error { err := createPersonFromIRI(ctx, repository.AttributedTo.GetLink()) if err != nil { return err } user, err := activitypub.PersonIRIToUser(ctx, repository.AttributedTo.GetLink()) if err != nil { return err } // Check if repo exists _, err = repo_model.GetRepositoryByOwnerAndNameCtx(ctx, user.Name, repository.Name.String()) if err == nil { return nil } repo, err := repo_service.CreateRepository(user, user, repo_module.CreateRepoOptions{ Name: repository.Name.String(), OriginalURL: repository.GetLink().String(), }) if err != nil { return err } if repository.ForkedFrom != nil { repo.IsFork = true forkedFrom, err := activitypub.RepositoryIRIToRepository(ctx, repository.ForkedFrom.GetLink()) if err != nil { return err } repo.ForkID = forkedFrom.ID } return nil } func createRepositoryFromIRI(ctx context.Context, repoIRI ap.IRI) error { repoURL, err := url.Parse(repoIRI.String()) if err != nil { return err } // Fetch repository object resp, err := activitypub.Fetch(repoURL) if err != nil { return err } // Parse repository object ap.ItemTyperFunc = forgefed.GetItemByType ap.JSONItemUnmarshal = forgefed.JSONUnmarshalerFn ap.NotEmptyChecker = forgefed.NotEmpty object, err := ap.UnmarshalJSON(resp) if err != nil { return err } // Create federated repo return forgefed.OnRepository(object, func(r *forgefed.Repository) error { return createRepository(ctx, r) }) } // Create a ticket func createTicket(ctx context.Context, ticket *forgefed.Ticket) error { if ticket.Origin != nil && ticket.Target != nil { return createPullRequest(ctx, ticket) } return createIssue(ctx, ticket) } // Create an issue func createIssue(ctx context.Context, ticket *forgefed.Ticket) error { err := createRepositoryFromIRI(ctx, ticket.Context.GetLink()) if err != nil { return err } // Construct issue user, err := activitypub.PersonIRIToUser(ctx, ap.IRI(ticket.AttributedTo.GetLink().String())) if err != nil { return err } repo, err := activitypub.RepositoryIRIToRepository(ctx, ap.IRI(ticket.Context.GetLink().String())) if err != nil { return err } idx, err := strconv.ParseInt(ticket.Name.String()[1:], 10, 64) if err != nil { return err } issue := &issues_model.Issue{ Index: idx, // This doesn't seem to work? RepoID: repo.ID, Repo: repo, Title: ticket.Summary.String(), PosterID: user.ID, Poster: user, Content: ticket.Content.String(), OriginalAuthor: ticket.GetLink().String(), // Create new database field to store IRI? IsClosed: ticket.IsResolved, } return issue_service.NewIssue(repo, issue, nil, nil, nil) } // Create a pull request func createPullRequest(ctx context.Context, ticket *forgefed.Ticket) error { err := createRepositoryFromIRI(ctx, ticket.Context.GetLink()) if err != nil { return err } user, err := activitypub.PersonIRIToUser(ctx, ticket.AttributedTo.GetLink()) if err != nil { return err } // Extract origin and target repos originUsername, originReponame, originBranch, err := activitypub.BranchIRIToName(ticket.Origin.GetLink()) if err != nil { return err } originRepo, err := repo_model.GetRepositoryByOwnerAndName(originUsername, originReponame) if err != nil { return err } targetUsername, targetReponame, targetBranch, err := activitypub.BranchIRIToName(ticket.Target.GetLink()) if err != nil { return err } targetRepo, err := repo_model.GetRepositoryByOwnerAndName(targetUsername, targetReponame) if err != nil { return err } idx, err := strconv.ParseInt(ticket.Name.String()[1:], 10, 64) if err != nil { return err } prIssue := &issues_model.Issue{ Index: idx, RepoID: targetRepo.ID, Title: ticket.Summary.String(), PosterID: user.ID, Poster: user, IsPull: true, Content: ticket.Content.String(), IsClosed: ticket.IsResolved, } pr := &issues_model.PullRequest{ HeadRepoID: originRepo.ID, BaseRepoID: targetRepo.ID, HeadBranch: originBranch, BaseBranch: targetBranch, HeadRepo: originRepo, BaseRepo: targetRepo, MergeBase: "", Type: issues_model.PullRequestGitea, } return pull_service.NewPullRequest(ctx, targetRepo, prIssue, []int64{}, []string{}, pr, []int64{}) } // Create a comment func createComment(ctx context.Context, note *ap.Note) error { err := createPersonFromIRI(ctx, note.AttributedTo.GetLink()) if err != nil { return err } user, err := activitypub.PersonIRIToUser(ctx, note.AttributedTo.GetLink()) if err != nil { return err } username, reponame, idx, err := activitypub.TicketIRIToName(note.Context.GetLink()) if err != nil { return err } repo, err := repo_model.GetRepositoryByOwnerAndNameCtx(ctx, username, reponame) if err != nil { return err } issue, err := issues_model.GetIssueByIndex(repo.ID, idx) if err != nil { return err } _, err = issues_model.CreateCommentCtx(ctx, &issues_model.CreateCommentOptions{ Doer: user, Repo: repo, Issue: issue, OldTitle: note.GetLink().String(), Content: note.Content.String(), }) return err }