Implement loading remote tickets
parent
20e8c64317
commit
19af0c9267
|
@ -6,12 +6,42 @@ package activitypub
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
issues_model "code.gitea.io/gitea/models/issues"
|
||||||
"code.gitea.io/gitea/modules/forgefed"
|
"code.gitea.io/gitea/modules/forgefed"
|
||||||
|
issue_service "code.gitea.io/gitea/services/issue"
|
||||||
|
|
||||||
|
ap "github.com/go-ap/activitypub"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Create an issue
|
// Create an issue
|
||||||
func ReceiveIssue(ctx context.Context, ticket *forgefed.Ticket) error {
|
func ReceiveIssue(ctx context.Context, ticket *forgefed.Ticket) error {
|
||||||
// TODO
|
// Construct issue
|
||||||
return nil
|
user, err := PersonIRIToUser(ctx, ap.IRI(ticket.AttributedTo.GetLink().String()))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
repo, err := RepositoryIRIToRepository(ctx, ap.IRI(ticket.Context.GetLink().String()))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Println(ticket)
|
||||||
|
fmt.Println(ticket.Name.String())
|
||||||
|
idx, err := strconv.ParseInt(ticket.Name.String()[1:], 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
issue := &issues_model.Issue{
|
||||||
|
ID: idx,
|
||||||
|
RepoID: repo.ID,
|
||||||
|
Repo: repo,
|
||||||
|
Title: ticket.Summary.String(),
|
||||||
|
PosterID: user.ID,
|
||||||
|
Poster: user,
|
||||||
|
Content: ticket.Content.String(),
|
||||||
|
}
|
||||||
|
fmt.Println(issue)
|
||||||
|
return issue_service.NewIssue(repo, issue, nil, nil, nil)
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,8 +15,7 @@ import (
|
||||||
|
|
||||||
// Create a new federated repo from a Repository object
|
// Create a new federated repo from a Repository object
|
||||||
func FederatedRepoNew(ctx context.Context, repository *forgefed.Repository) error {
|
func FederatedRepoNew(ctx context.Context, repository *forgefed.Repository) error {
|
||||||
ownerIRI := repository.AttributedTo.GetLink()
|
user, err := PersonIRIToUser(ctx, repository.AttributedTo.GetLink())
|
||||||
user, err := PersonIRIToUser(ctx, ownerIRI)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,6 +46,7 @@ func Ticket(ctx *context.APIContext) {
|
||||||
link := setting.AppURL + "api/v1/activitypub/ticket/" + ctx.ContextUser.Name + "/" + ctx.Repo.Repository.Name + "/" + ctx.Params("id")
|
link := setting.AppURL + "api/v1/activitypub/ticket/" + ctx.ContextUser.Name + "/" + ctx.Repo.Repository.Name + "/" + ctx.Params("id")
|
||||||
|
|
||||||
ticket := forgefed.TicketNew()
|
ticket := forgefed.TicketNew()
|
||||||
|
ticket.Type = forgefed.TicketType
|
||||||
ticket.ID = ap.IRI(link)
|
ticket.ID = ap.IRI(link)
|
||||||
|
|
||||||
repo, err := repo_model.GetRepositoryByOwnerAndNameCtx(ctx, ctx.ContextUser.Name, ctx.Repo.Repository.Name)
|
repo, err := repo_model.GetRepositoryByOwnerAndNameCtx(ctx, ctx.ContextUser.Name, ctx.Repo.Repository.Name)
|
||||||
|
@ -64,7 +65,16 @@ func Ticket(ctx *context.APIContext) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ticket.Context = ap.IRI(setting.AppURL + ctx.ContextUser.Name + "/" + ctx.Repo.Repository.Name)
|
ticket.Name = ap.NaturalLanguageValuesNew()
|
||||||
|
// Setting a NaturalLanguageValue to a number causes go-ap's JSON parsing to do weird things
|
||||||
|
// Workaround: set it to #1 instead of 1
|
||||||
|
err = ticket.Name.Set("en", ap.Content("#" + ctx.Params("id")))
|
||||||
|
if err != nil {
|
||||||
|
ctx.ServerError("Set Name", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ticket.Context = ap.IRI(setting.AppURL + "api/v1/activitypub/repo/" + ctx.ContextUser.Name + "/" + ctx.Repo.Repository.Name)
|
||||||
|
|
||||||
err = issue.LoadPoster()
|
err = issue.LoadPoster()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -7,6 +7,7 @@ package web
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/activitypub"
|
"code.gitea.io/gitea/modules/activitypub"
|
||||||
"code.gitea.io/gitea/modules/context"
|
"code.gitea.io/gitea/modules/context"
|
||||||
|
@ -19,7 +20,7 @@ import (
|
||||||
func AuthorizeInteraction(ctx *context.Context) {
|
func AuthorizeInteraction(ctx *context.Context) {
|
||||||
uri, err := url.Parse(ctx.Req.URL.Query().Get("uri"))
|
uri, err := url.Parse(ctx.Req.URL.Query().Get("uri"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("Could not parse URI", err)
|
ctx.ServerError("Parse URI", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
resp, err := activitypub.Fetch(uri)
|
resp, err := activitypub.Fetch(uri)
|
||||||
|
@ -27,7 +28,7 @@ func AuthorizeInteraction(ctx *context.Context) {
|
||||||
ctx.ServerError("Fetch", err)
|
ctx.ServerError("Fetch", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ap.ItemTyperFunc = forgefed.GetItemByType
|
ap.ItemTyperFunc = forgefed.GetItemByType
|
||||||
ap.JSONItemUnmarshal = forgefed.JSONUnmarshalerFn
|
ap.JSONItemUnmarshal = forgefed.JSONUnmarshalerFn
|
||||||
object, err := ap.UnmarshalJSON(resp)
|
object, err := ap.UnmarshalJSON(resp)
|
||||||
|
@ -35,9 +36,10 @@ func AuthorizeInteraction(ctx *context.Context) {
|
||||||
ctx.ServerError("UnmarshalJSON", err)
|
ctx.ServerError("UnmarshalJSON", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
switch object.GetType() {
|
switch object.GetType() {
|
||||||
case ap.PersonType:
|
case ap.PersonType:
|
||||||
|
// Federated user
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("UnmarshalJSON", err)
|
ctx.ServerError("UnmarshalJSON", err)
|
||||||
return
|
return
|
||||||
|
@ -49,12 +51,34 @@ func AuthorizeInteraction(ctx *context.Context) {
|
||||||
}
|
}
|
||||||
name, err := activitypub.PersonIRIToName(object.GetLink())
|
name, err := activitypub.PersonIRIToName(object.GetLink())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("personIRIToName", err)
|
ctx.ServerError("PersonIRIToName", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
ctx.Redirect(name)
|
ctx.Redirect(name)
|
||||||
case forgefed.RepositoryType:
|
case forgefed.RepositoryType:
|
||||||
|
// Federated repository
|
||||||
err = forgefed.OnRepository(object, func(r *forgefed.Repository) error {
|
err = forgefed.OnRepository(object, func(r *forgefed.Repository) error {
|
||||||
|
ownerURL, err := url.Parse(r.AttributedTo.GetLink().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
|
||||||
|
object, err := ap.UnmarshalJSON(resp)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Create federated user
|
||||||
|
err = user_service.FederatedUserNew(ctx, object.(*ap.Person))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
return activitypub.FederatedRepoNew(ctx, r)
|
return activitypub.FederatedRepoNew(ctx, r)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -63,19 +87,53 @@ func AuthorizeInteraction(ctx *context.Context) {
|
||||||
}
|
}
|
||||||
username, reponame, err := activitypub.RepositoryIRIToName(object.GetLink())
|
username, reponame, err := activitypub.RepositoryIRIToName(object.GetLink())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("repositoryIRIToName", err)
|
ctx.ServerError("RepositoryIRIToName", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
ctx.Redirect(username + "/" + reponame)
|
ctx.Redirect(username + "/" + reponame)
|
||||||
case forgefed.TicketType:
|
case forgefed.TicketType:
|
||||||
|
// Federated ticket
|
||||||
err = forgefed.OnTicket(object, func(t *forgefed.Ticket) error {
|
err = forgefed.OnTicket(object, func(t *forgefed.Ticket) error {
|
||||||
|
// TODO: make sure federated user exists
|
||||||
|
// Also, refactor this code to reduce the chance of accidentally creating import cycles
|
||||||
|
repoURL, err := url.Parse(t.Context.GetLink().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
|
||||||
|
object, err := ap.UnmarshalJSON(resp)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Create federated repo
|
||||||
|
err = forgefed.OnRepository(object, func(r *forgefed.Repository) error {
|
||||||
|
return activitypub.FederatedRepoNew(ctx, r)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
return activitypub.ReceiveIssue(ctx, t)
|
return activitypub.ReceiveIssue(ctx, t)
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.ServerError("ReceiveIssue", err)
|
ctx.ServerError("ReceiveIssue", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// TODO: Implement ticketIRIToName and redirect to ticket
|
username, reponame, idx, err := activitypub.TicketIRIToName(object.GetLink())
|
||||||
|
if err != nil {
|
||||||
|
ctx.ServerError("TicketIRIToName", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(username + "/" + reponame + "/issues/" + strconv.FormatInt(idx, 10))
|
||||||
|
default:
|
||||||
|
ctx.ServerError("Not implemented", err)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.Status(http.StatusOK)
|
ctx.Status(http.StatusOK)
|
||||||
|
|
Loading…
Reference in New Issue