aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--appservice/api/query.go4
-rw-r--r--clientapi/auth/authtypes/profile.go2
-rw-r--r--clientapi/routing/profile.go107
-rw-r--r--clientapi/routing/routing.go6
-rw-r--r--common/types.go5
-rw-r--r--testfile1
6 files changed, 100 insertions, 25 deletions
diff --git a/appservice/api/query.go b/appservice/api/query.go
index 8ce3b4e0..9542df56 100644
--- a/appservice/api/query.go
+++ b/appservice/api/query.go
@@ -20,13 +20,13 @@ package api
import (
"context"
"database/sql"
- "errors"
"net/http"
"github.com/matrix-org/dendrite/clientapi/auth/authtypes"
"github.com/matrix-org/dendrite/clientapi/auth/storage/accounts"
"github.com/matrix-org/gomatrixserverlib"
+ "github.com/matrix-org/dendrite/common"
commonHTTP "github.com/matrix-org/dendrite/common/http"
opentracing "github.com/opentracing/opentracing-go"
)
@@ -164,7 +164,7 @@ func RetrieveUserProfile(
// If no user exists, return
if !userResp.UserIDExists {
- return nil, errors.New("no known profile for given user ID")
+ return nil, common.ErrProfileNoExists
}
// Try to query the user from the local database again
diff --git a/clientapi/auth/authtypes/profile.go b/clientapi/auth/authtypes/profile.go
index 6cf508f4..0bc49658 100644
--- a/clientapi/auth/authtypes/profile.go
+++ b/clientapi/auth/authtypes/profile.go
@@ -14,7 +14,7 @@
package authtypes
-// Profile represents the profile for a Matrix account on this home server.
+// Profile represents the profile for a Matrix account.
type Profile struct {
Localpart string
DisplayName string
diff --git a/clientapi/routing/profile.go b/clientapi/routing/profile.go
index 8d28b366..e8ea6cf1 100644
--- a/clientapi/routing/profile.go
+++ b/clientapi/routing/profile.go
@@ -30,43 +30,61 @@ import (
"github.com/matrix-org/dendrite/roomserver/api"
"github.com/matrix-org/gomatrixserverlib"
+ "github.com/matrix-org/gomatrix"
"github.com/matrix-org/util"
)
// GetProfile implements GET /profile/{userID}
func GetProfile(
- req *http.Request, accountDB *accounts.Database, userID string, asAPI appserviceAPI.AppServiceQueryAPI,
+ req *http.Request, accountDB *accounts.Database, cfg *config.Dendrite,
+ userID string,
+ asAPI appserviceAPI.AppServiceQueryAPI,
+ federation *gomatrixserverlib.FederationClient,
) util.JSONResponse {
- profile, err := appserviceAPI.RetrieveUserProfile(req.Context(), userID, asAPI, accountDB)
+ profile, err := getProfile(req.Context(), accountDB, cfg, userID, asAPI, federation)
if err != nil {
+ if err == common.ErrProfileNoExists {
+ return util.JSONResponse{
+ Code: http.StatusNotFound,
+ JSON: jsonerror.NotFound("The user does not exist or does not have a profile"),
+ }
+ }
+
return httputil.LogThenError(req, err)
}
- res := common.ProfileResponse{
- AvatarURL: profile.AvatarURL,
- DisplayName: profile.DisplayName,
- }
return util.JSONResponse{
Code: http.StatusOK,
- JSON: res,
+ JSON: common.ProfileResponse{
+ AvatarURL: profile.AvatarURL,
+ DisplayName: profile.DisplayName,
+ },
}
}
// GetAvatarURL implements GET /profile/{userID}/avatar_url
func GetAvatarURL(
- req *http.Request, accountDB *accounts.Database, userID string, asAPI appserviceAPI.AppServiceQueryAPI,
+ req *http.Request, accountDB *accounts.Database, cfg *config.Dendrite,
+ userID string, asAPI appserviceAPI.AppServiceQueryAPI,
+ federation *gomatrixserverlib.FederationClient,
) util.JSONResponse {
- profile, err := appserviceAPI.RetrieveUserProfile(req.Context(), userID, asAPI, accountDB)
+ profile, err := getProfile(req.Context(), accountDB, cfg, userID, asAPI, federation)
if err != nil {
+ if err == common.ErrProfileNoExists {
+ return util.JSONResponse{
+ Code: http.StatusNotFound,
+ JSON: jsonerror.NotFound("The user does not exist or does not have a profile"),
+ }
+ }
+
return httputil.LogThenError(req, err)
}
- res := common.AvatarURL{
- AvatarURL: profile.AvatarURL,
- }
return util.JSONResponse{
Code: http.StatusOK,
- JSON: res,
+ JSON: common.AvatarURL{
+ AvatarURL: profile.AvatarURL,
+ },
}
}
@@ -152,18 +170,27 @@ func SetAvatarURL(
// GetDisplayName implements GET /profile/{userID}/displayname
func GetDisplayName(
- req *http.Request, accountDB *accounts.Database, userID string, asAPI appserviceAPI.AppServiceQueryAPI,
+ req *http.Request, accountDB *accounts.Database, cfg *config.Dendrite,
+ userID string, asAPI appserviceAPI.AppServiceQueryAPI,
+ federation *gomatrixserverlib.FederationClient,
) util.JSONResponse {
- profile, err := appserviceAPI.RetrieveUserProfile(req.Context(), userID, asAPI, accountDB)
+ profile, err := getProfile(req.Context(), accountDB, cfg, userID, asAPI, federation)
if err != nil {
+ if err == common.ErrProfileNoExists {
+ return util.JSONResponse{
+ Code: http.StatusNotFound,
+ JSON: jsonerror.NotFound("The user does not exist or does not have a profile"),
+ }
+ }
+
return httputil.LogThenError(req, err)
}
- res := common.DisplayName{
- DisplayName: profile.DisplayName,
- }
+
return util.JSONResponse{
Code: http.StatusOK,
- JSON: res,
+ JSON: common.DisplayName{
+ DisplayName: profile.DisplayName,
+ },
}
}
@@ -247,6 +274,48 @@ func SetDisplayName(
}
}
+// getProfile gets the full profile of a user by querying the database or a
+// remote homeserver.
+// Returns an error when something goes wrong or specifically
+// common.ErrProfileNoExists when the profile doesn't exist.
+func getProfile(
+ ctx context.Context, accountDB *accounts.Database, cfg *config.Dendrite,
+ userID string,
+ asAPI appserviceAPI.AppServiceQueryAPI,
+ federation *gomatrixserverlib.FederationClient,
+) (*authtypes.Profile, error) {
+ localpart, domain, err := gomatrixserverlib.SplitID('@', userID)
+ if err != nil {
+ return nil, err
+ }
+
+ if domain != cfg.Matrix.ServerName {
+ profile, fedErr := federation.LookupProfile(ctx, domain, userID, "")
+ if fedErr != nil {
+ if x, ok := fedErr.(gomatrix.HTTPError); ok {
+ if x.Code == http.StatusNotFound {
+ return nil, common.ErrProfileNoExists
+ }
+ }
+
+ return nil, fedErr
+ }
+
+ return &authtypes.Profile{
+ Localpart: localpart,
+ DisplayName: profile.DisplayName,
+ AvatarURL: profile.AvatarURL,
+ }, nil
+ }
+
+ profile, err := appserviceAPI.RetrieveUserProfile(ctx, userID, asAPI, accountDB)
+ if err != nil {
+ return nil, err
+ }
+
+ return profile, nil
+}
+
func buildMembershipEvents(
ctx context.Context,
memberships []authtypes.Membership,
diff --git a/clientapi/routing/routing.go b/clientapi/routing/routing.go
index c262db3d..825dd97a 100644
--- a/clientapi/routing/routing.go
+++ b/clientapi/routing/routing.go
@@ -283,7 +283,7 @@ func Setup(
if err != nil {
return util.ErrorResponse(err)
}
- return GetProfile(req, accountDB, vars["userID"], asAPI)
+ return GetProfile(req, accountDB, &cfg, vars["userID"], asAPI, federation)
}),
).Methods(http.MethodGet, http.MethodOptions)
@@ -293,7 +293,7 @@ func Setup(
if err != nil {
return util.ErrorResponse(err)
}
- return GetAvatarURL(req, accountDB, vars["userID"], asAPI)
+ return GetAvatarURL(req, accountDB, &cfg, vars["userID"], asAPI, federation)
}),
).Methods(http.MethodGet, http.MethodOptions)
@@ -315,7 +315,7 @@ func Setup(
if err != nil {
return util.ErrorResponse(err)
}
- return GetDisplayName(req, accountDB, vars["userID"], asAPI)
+ return GetDisplayName(req, accountDB, &cfg, vars["userID"], asAPI, federation)
}),
).Methods(http.MethodGet, http.MethodOptions)
diff --git a/common/types.go b/common/types.go
index 6888d380..91765be0 100644
--- a/common/types.go
+++ b/common/types.go
@@ -15,9 +15,14 @@
package common
import (
+ "errors"
"strconv"
)
+// ErrProfileNoExists is returned when trying to lookup a user's profile that
+// doesn't exist locally.
+var ErrProfileNoExists = errors.New("no known profile for given user ID")
+
// AccountData represents account data sent from the client API server to the
// sync API server
type AccountData struct {
diff --git a/testfile b/testfile
index 5791938f..74c9d9e4 100644
--- a/testfile
+++ b/testfile
@@ -167,3 +167,4 @@ Tags appear in an initial v2 /sync
Newly updated tags appear in an incremental v2 /sync
Deleted tags appear in an incremental v2 /sync
/event/ on non world readable room does not work
+Outbound federation can query profile data