2023-11-16 08:49:05 -05:00
package activitypub
import (
"fmt"
"net/url"
2023-11-23 11:03:24 -05:00
"strconv"
2023-11-16 08:49:05 -05:00
"strings"
2023-11-24 06:48:14 -05:00
2023-12-06 07:06:30 -05:00
"code.gitea.io/gitea/modules/log"
2023-12-08 12:08:54 -05:00
"code.gitea.io/gitea/modules/validation"
2023-11-16 08:49:05 -05:00
)
2023-11-23 08:50:32 -05:00
type Validatable interface { // ToDo: What is the right package for this interface?
2023-11-24 05:37:29 -05:00
validate_is_not_nil ( ) error
2023-11-22 10:08:14 -05:00
validate_is_not_empty ( ) error
2023-11-22 09:25:43 -05:00
Validate ( ) error
2023-11-24 05:37:29 -05:00
IsValid ( ) ( bool , error )
PanicIfInvalid ( )
2023-11-22 09:25:43 -05:00
}
2023-12-08 12:33:26 -05:00
type ActorId struct {
2023-11-16 08:49:05 -05:00
userId string
2023-12-06 09:15:39 -05:00
source string
2023-11-24 06:49:36 -05:00
schema string
2023-11-16 08:49:05 -05:00
path string
host string
port string // optional
}
2023-12-07 05:24:27 -05:00
func validate_is_not_empty ( str string ) error {
2023-11-24 05:37:29 -05:00
if str == "" {
2023-12-07 05:24:27 -05:00
return fmt . Errorf ( "the given string was empty" )
2023-11-24 05:37:29 -05:00
}
return nil
}
/ *
2023-12-06 05:24:42 -05:00
Validate collects error strings in a slice and returns this
2023-11-24 05:37:29 -05:00
* /
2023-12-08 12:33:26 -05:00
func ( a ActorId ) Validate ( ) [ ] string {
2023-11-16 08:49:05 -05:00
2023-12-07 05:24:47 -05:00
var err = [ ] string { }
2023-11-24 05:37:29 -05:00
2023-12-08 12:08:54 -05:00
if res := validation . ValidateNotEmpty ( a . schema , "schema" ) ; res != nil {
err = append ( err , res . Error ( ) )
2023-11-22 09:27:44 -05:00
}
2023-12-07 05:24:47 -05:00
if res := validate_is_not_empty ( a . host ) ; res != nil {
err = append ( err , strings . Join ( [ ] string { res . Error ( ) , "for host field" } , " " ) )
2023-11-16 08:49:05 -05:00
}
2023-11-24 06:48:14 -05:00
switch a . source {
case "forgejo" , "gitea" :
2023-12-06 09:16:01 -05:00
if ! strings . Contains ( a . path , "api/v1/activitypub/user-id" ) &&
! strings . Contains ( a . path , "api/v1/activitypub/repository-id" ) {
err = append ( err , fmt . Errorf ( "the Path to the API was invalid: ---%v---" , a . path ) . Error ( ) )
2023-11-24 06:48:14 -05:00
}
default :
err = append ( err , fmt . Errorf ( "currently only forgeo and gitea sources are allowed from actor id" ) . Error ( ) )
2023-11-16 08:49:05 -05:00
}
2023-11-24 05:37:29 -05:00
return err
2023-11-16 08:49:05 -05:00
}
2023-12-06 05:24:42 -05:00
/ *
IsValid concatenates the error messages with newlines and returns them if there are any
* /
2023-12-08 12:33:26 -05:00
func ( a ActorId ) IsValid ( ) ( bool , error ) {
2023-11-24 05:37:29 -05:00
if err := a . Validate ( ) ; len ( err ) > 0 {
errString := strings . Join ( err , "\n" )
return false , fmt . Errorf ( errString )
}
return true , nil
}
2023-12-08 12:33:26 -05:00
func ( a ActorId ) PanicIfInvalid ( ) {
2023-11-24 05:37:29 -05:00
if valid , err := a . IsValid ( ) ; ! valid {
panic ( err )
}
}
2023-12-08 12:33:26 -05:00
func ( a ActorId ) GetUserId ( ) int {
2023-11-24 06:49:36 -05:00
result , err := strconv . Atoi ( a . userId )
if err != nil {
panic ( err )
}
return result
}
2023-12-08 12:33:26 -05:00
func ( a ActorId ) GetNormalizedUri ( ) string {
2023-12-06 03:07:09 -05:00
result := fmt . Sprintf ( "%s://%s:%s/%s/%s" , a . schema , a . host , a . port , a . path , a . userId )
return result
}
2023-11-24 06:49:36 -05:00
// Returns the combination of host:port if port exists, host otherwise
2023-12-08 12:33:26 -05:00
func ( a ActorId ) GetHostAndPort ( ) string {
2023-11-24 06:49:36 -05:00
if a . port != "" {
return strings . Join ( [ ] string { a . host , a . port } , ":" )
}
return a . host
}
2023-12-06 07:06:30 -05:00
func containsEmptyString ( ar [ ] string ) bool {
for _ , elem := range ar {
if elem == "" {
return true
}
}
return false
}
func removeEmptyStrings ( ls [ ] string ) [ ] string {
var rs [ ] string
for _ , str := range ls {
if str != "" {
rs = append ( rs , str )
}
}
return rs
}
2023-12-08 05:54:07 -05:00
func ValidateAndParseIRI ( unvalidatedIRI string ) ( url . URL , error ) { // ToDo: Validate that it is not the same host as ours.
2023-12-07 05:44:59 -05:00
err := validate_is_not_empty ( unvalidatedIRI ) // url.Parse seems to accept empty strings?
if err != nil {
return url . URL { } , err
2023-12-06 10:14:39 -05:00
}
2023-12-07 05:44:59 -05:00
validatedURL , err := url . Parse ( unvalidatedIRI )
2023-11-16 08:49:05 -05:00
if err != nil {
2023-12-07 05:44:59 -05:00
return url . URL { } , err
2023-11-16 08:49:05 -05:00
}
2023-12-07 06:03:28 -05:00
if len ( validatedURL . Path ) <= 1 {
return url . URL { } , fmt . Errorf ( "path was empty" )
}
2023-12-07 05:44:59 -05:00
return * validatedURL , nil
}
// TODO: This parsing is very Person-Specific. We should adjust the name & move to a better location (maybe forgefed package?)
2023-12-08 12:33:26 -05:00
func ParseActorID ( validatedURL url . URL , source string ) ActorId { // ToDo: Turn this into a factory function and do not split parsing and validation rigurously
2023-12-07 05:44:59 -05:00
pathWithUserID := strings . Split ( validatedURL . Path , "/" )
2023-12-06 07:06:30 -05:00
if containsEmptyString ( pathWithUserID ) {
pathWithUserID = removeEmptyStrings ( pathWithUserID )
}
length := len ( pathWithUserID )
pathWithoutUserID := strings . Join ( pathWithUserID [ 0 : length - 1 ] , "/" )
userId := pathWithUserID [ length - 1 ]
log . Info ( "Actor: pathWithUserID: %s" , pathWithUserID )
log . Info ( "Actor: pathWithoutUserID: %s" , pathWithoutUserID )
log . Info ( "Actor: UserID: %s" , userId )
2023-11-16 08:49:05 -05:00
2023-12-08 12:33:26 -05:00
return ActorId { // ToDo: maybe keep original input to validate against (maybe extra method)
2023-11-16 08:49:05 -05:00
userId : userId ,
2023-12-06 09:15:39 -05:00
source : source ,
2023-12-07 05:44:59 -05:00
schema : validatedURL . Scheme ,
host : validatedURL . Hostname ( ) , // u.Host returns hostname:port
2023-12-06 07:06:30 -05:00
path : pathWithoutUserID ,
2023-12-07 05:44:59 -05:00
port : validatedURL . Port ( ) ,
}
2023-11-16 08:49:05 -05:00
}
2023-12-08 12:08:54 -05:00
2023-12-08 12:33:26 -05:00
func NewActorId ( uri string , source string ) ( ActorId , error ) {
2023-12-08 12:08:54 -05:00
if ! validation . IsValidExternalURL ( uri ) {
2023-12-08 12:33:26 -05:00
return ActorId { } , fmt . Errorf ( "uri %s is not a valid external url" , uri )
2023-12-08 12:08:54 -05:00
}
2023-12-08 12:33:26 -05:00
return ActorId { } , nil
2023-12-08 12:08:54 -05:00
}