2017-08-21 16:34:26 +01:00
|
|
|
// Copyright 2017 Vector Creations Ltd
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2017-10-11 18:16:53 +01:00
|
|
|
package routing
|
2017-08-21 16:34:26 +01:00
|
|
|
|
|
|
|
import (
|
2020-06-17 16:21:42 +01:00
|
|
|
"encoding/json"
|
2023-01-17 09:08:23 +00:00
|
|
|
"math"
|
2017-08-21 16:34:26 +01:00
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/matrix-org/dendrite/roomserver/api"
|
2022-10-25 11:39:10 +01:00
|
|
|
"github.com/matrix-org/dendrite/syncapi/storage"
|
2023-04-04 18:16:53 +01:00
|
|
|
"github.com/matrix-org/dendrite/syncapi/synctypes"
|
2022-10-25 11:39:10 +01:00
|
|
|
"github.com/matrix-org/dendrite/syncapi/types"
|
2020-06-16 14:10:55 +01:00
|
|
|
userapi "github.com/matrix-org/dendrite/userapi/api"
|
2023-04-28 16:00:22 +01:00
|
|
|
"github.com/matrix-org/gomatrixserverlib"
|
2023-05-09 23:46:49 +01:00
|
|
|
"github.com/matrix-org/gomatrixserverlib/spec"
|
2023-01-17 09:08:23 +00:00
|
|
|
"github.com/matrix-org/util"
|
2017-08-21 16:34:26 +01:00
|
|
|
)
|
|
|
|
|
2020-03-19 10:25:36 +00:00
|
|
|
type getMembershipResponse struct {
|
2023-04-04 18:16:53 +01:00
|
|
|
Chunk []synctypes.ClientEvent `json:"chunk"`
|
2017-08-24 16:00:14 +01:00
|
|
|
}
|
|
|
|
|
2020-06-17 16:21:42 +01:00
|
|
|
// https://matrix.org/docs/spec/client_server/r0.6.0#get-matrix-client-r0-rooms-roomid-joined-members
|
|
|
|
type getJoinedMembersResponse struct {
|
|
|
|
Joined map[string]joinedMember `json:"joined"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type joinedMember struct {
|
|
|
|
DisplayName string `json:"display_name"`
|
|
|
|
AvatarURL string `json:"avatar_url"`
|
|
|
|
}
|
|
|
|
|
2020-12-03 11:01:49 +00:00
|
|
|
// The database stores 'displayname' without an underscore.
|
|
|
|
// Deserialize into this and then change to the actual API response
|
|
|
|
type databaseJoinedMember struct {
|
|
|
|
DisplayName string `json:"displayname"`
|
|
|
|
AvatarURL string `json:"avatar_url"`
|
|
|
|
}
|
|
|
|
|
2022-10-25 11:39:10 +01:00
|
|
|
// GetMemberships implements
|
|
|
|
//
|
|
|
|
// GET /rooms/{roomId}/members
|
|
|
|
// GET /rooms/{roomId}/joined_members
|
2017-08-21 16:34:26 +01:00
|
|
|
func GetMemberships(
|
2022-10-25 11:39:10 +01:00
|
|
|
req *http.Request, device *userapi.Device, roomID string,
|
|
|
|
syncDB storage.Database, rsAPI api.SyncRoomserverAPI,
|
|
|
|
joinedOnly bool, membership, notMembership *string, at string,
|
2017-08-21 16:34:26 +01:00
|
|
|
) util.JSONResponse {
|
2023-06-12 12:19:25 +01:00
|
|
|
userID, err := spec.NewUserID(device.UserID, true)
|
|
|
|
if err != nil {
|
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusBadRequest,
|
|
|
|
JSON: spec.InvalidParam("Device UserID is invalid"),
|
|
|
|
}
|
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
queryReq := api.QueryMembershipForUserRequest{
|
|
|
|
RoomID: roomID,
|
2023-06-12 12:19:25 +01:00
|
|
|
UserID: *userID,
|
2017-08-21 16:34:26 +01:00
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
|
|
|
|
var queryRes api.QueryMembershipForUserResponse
|
2023-06-12 12:19:25 +01:00
|
|
|
if queryErr := rsAPI.QueryMembershipForUser(req.Context(), &queryReq, &queryRes); queryErr != nil {
|
|
|
|
util.GetLogger(req.Context()).WithError(queryErr).Error("rsAPI.QueryMembershipsForRoom failed")
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2017-08-21 16:34:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if !queryRes.HasBeenInRoom {
|
|
|
|
return util.JSONResponse{
|
2018-03-13 15:55:45 +00:00
|
|
|
Code: http.StatusForbidden,
|
2023-05-09 23:46:49 +01:00
|
|
|
JSON: spec.Forbidden("You aren't a member of the room and weren't previously a member of the room."),
|
2017-08-21 16:34:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-27 13:18:22 +01:00
|
|
|
if joinedOnly && !queryRes.IsInRoom {
|
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusForbidden,
|
2023-05-09 23:46:49 +01:00
|
|
|
JSON: spec.Forbidden("You aren't a member of the room and weren't previously a member of the room."),
|
2022-10-27 13:18:22 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-25 11:39:10 +01:00
|
|
|
db, err := syncDB.NewDatabaseSnapshot(req.Context())
|
|
|
|
if err != nil {
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
}
|
2023-01-17 09:08:23 +00:00
|
|
|
defer db.Rollback() // nolint: errcheck
|
2022-10-25 11:39:10 +01:00
|
|
|
|
|
|
|
atToken, err := types.NewTopologyTokenFromString(at)
|
|
|
|
if err != nil {
|
2023-01-17 09:08:23 +00:00
|
|
|
atToken = types.TopologyToken{Depth: math.MaxInt64, PDUPosition: math.MaxInt64}
|
2022-10-25 11:39:10 +01:00
|
|
|
if queryRes.HasBeenInRoom && !queryRes.IsInRoom {
|
|
|
|
// If you have left the room then this will be the members of the room when you left.
|
|
|
|
atToken, err = db.EventPositionInTopology(req.Context(), queryRes.EventID)
|
2023-01-17 09:08:23 +00:00
|
|
|
if err != nil {
|
|
|
|
util.GetLogger(req.Context()).WithError(err).Error("unable to get 'atToken'")
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2023-01-17 09:08:23 +00:00
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
eventIDs, err := db.SelectMemberships(req.Context(), roomID, atToken, membership, notMembership)
|
|
|
|
if err != nil {
|
|
|
|
util.GetLogger(req.Context()).WithError(err).Error("db.SelectMemberships failed")
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
}
|
|
|
|
|
2022-10-27 13:18:22 +01:00
|
|
|
qryRes := &api.QueryEventsByIDResponse{}
|
2023-03-01 16:06:47 +00:00
|
|
|
if err := rsAPI.QueryEventsByID(req.Context(), &api.QueryEventsByIDRequest{EventIDs: eventIDs, RoomID: roomID}, qryRes); err != nil {
|
2022-10-27 13:18:22 +01:00
|
|
|
util.GetLogger(req.Context()).WithError(err).Error("rsAPI.QueryEventsByID failed")
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2022-10-25 11:39:10 +01:00
|
|
|
}
|
|
|
|
|
2022-10-27 13:18:22 +01:00
|
|
|
result := qryRes.Events
|
|
|
|
|
2020-06-17 16:21:42 +01:00
|
|
|
if joinedOnly {
|
|
|
|
var res getJoinedMembersResponse
|
|
|
|
res.Joined = make(map[string]joinedMember)
|
2022-10-25 11:39:10 +01:00
|
|
|
for _, ev := range result {
|
2020-12-03 11:01:49 +00:00
|
|
|
var content databaseJoinedMember
|
2022-10-25 11:39:10 +01:00
|
|
|
if err := json.Unmarshal(ev.Content(), &content); err != nil {
|
2020-06-17 16:21:42 +01:00
|
|
|
util.GetLogger(req.Context()).WithError(err).Error("failed to unmarshal event content")
|
2023-05-17 01:33:27 +01:00
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
2020-06-17 16:21:42 +01:00
|
|
|
}
|
2023-06-07 18:14:35 +01:00
|
|
|
|
2023-09-15 15:39:06 +01:00
|
|
|
userID, err := rsAPI.QueryUserIDForSender(req.Context(), ev.RoomID(), ev.SenderID())
|
2023-06-07 18:14:35 +01:00
|
|
|
if err != nil || userID == nil {
|
|
|
|
util.GetLogger(req.Context()).WithError(err).Error("rsAPI.QueryUserIDForSender failed")
|
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusInternalServerError,
|
|
|
|
JSON: spec.InternalServerError{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusForbidden,
|
|
|
|
JSON: spec.Forbidden("You don't have permission to kick this user, unknown senderID"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
res.Joined[userID.String()] = joinedMember(content)
|
2020-06-17 16:21:42 +01:00
|
|
|
}
|
|
|
|
return util.JSONResponse{
|
|
|
|
Code: http.StatusOK,
|
|
|
|
JSON: res,
|
|
|
|
}
|
|
|
|
}
|
2017-08-21 16:34:26 +01:00
|
|
|
return util.JSONResponse{
|
2018-03-13 15:55:45 +00:00
|
|
|
Code: http.StatusOK,
|
2023-06-14 15:23:46 +01:00
|
|
|
JSON: getMembershipResponse{synctypes.ToClientEvents(gomatrixserverlib.ToPDUs(result), synctypes.FormatAll, func(roomID spec.RoomID, senderID spec.SenderID) (*spec.UserID, error) {
|
2023-06-06 21:55:18 +01:00
|
|
|
return rsAPI.QueryUserIDForSender(req.Context(), roomID, senderID)
|
|
|
|
})},
|
2017-08-21 16:34:26 +01:00
|
|
|
}
|
|
|
|
}
|