-
Notifications
You must be signed in to change notification settings - Fork 52
/
Copy pathperforminvite.go
291 lines (249 loc) · 10.2 KB
/
performinvite.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
// Copyright 2023 The Matrix.org Foundation C.I.C.
//
// 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.
package gomatrixserverlib
import (
"context"
"crypto/ed25519"
"fmt"
"time"
"github.com/matrix-org/gomatrixserverlib/spec"
"github.com/sirupsen/logrus"
)
type GetLatestEvents func(ctx context.Context, roomID spec.RoomID, eventsNeeded []StateKeyTuple) (LatestEvents, error)
type PerformInviteInput struct {
RoomID spec.RoomID // The room the user is being invited to join
RoomVersion RoomVersion
Inviter spec.UserID // The user doing the inviting
Invitee spec.UserID // The user being invited join the room
IsTargetLocal bool // Whether the user being invited is local to this server
EventTemplate ProtoEvent // The original invite event
StrippedState []InviteStrippedState // A small set of state events that can be used to identify the room
KeyID KeyID
SigningKey ed25519.PrivateKey
EventTime time.Time
MembershipQuerier MembershipQuerier // Provides information about the room's membership
StateQuerier StateQuerier // Provides access to state events
UserIDQuerier spec.UserIDForSender // Provides userID for a given senderID
SenderIDQuerier spec.SenderIDForUser // Provides senderID for a given userID
SenderIDCreator spec.CreateSenderID
EventQuerier GetLatestEvents
StoreSenderIDFromPublicID spec.StoreSenderIDFromPublicID // Creates the senderID -> userID for the room creator
}
// PerformInvite - Performs all the checks required to validate the invite is allowed
// to happen.
// On success will return either nothing (in the case of inviting a local user) or
// a fully formed & signed Invite Event (in the case of inviting a remote user)
// nolint:gocyclo
func PerformInvite(ctx context.Context, input PerformInviteInput, fedClient FederatedInviteClient) (PDU, error) {
if input.MembershipQuerier == nil || input.StateQuerier == nil || input.UserIDQuerier == nil ||
input.SenderIDQuerier == nil || input.SenderIDCreator == nil || input.EventQuerier == nil {
panic("Missing valid Querier")
}
if ctx == nil {
panic("Missing valid Context")
}
logger := createInviteLogger(ctx, input.RoomID, input.Inviter, input.Invitee, "")
logger.WithFields(logrus.Fields{
"room_version": input.RoomVersion,
"target_local": input.IsTargetLocal,
"origin_local": true,
}).Debug("processing invite event")
inviteState := input.StrippedState
if len(inviteState) == 0 {
var err error
inviteState, err = GenerateStrippedState(ctx, input.RoomID, input.StateQuerier)
if err != nil {
logger.WithError(err).Error("failed generating stripped state")
return nil, spec.InternalServerError{}
}
}
err := setUnsignedFieldForProtoInvite(&input.EventTemplate, inviteState)
if err != nil {
return nil, err
}
// Check that we can accept invites for this room version.
verImpl, err := GetRoomVersion(input.RoomVersion)
if err != nil {
return nil, spec.UnsupportedRoomVersion(
fmt.Sprintf("Room version %q is not supported by this server.", input.RoomVersion),
)
}
invitedSenderID, err := input.SenderIDQuerier(input.RoomID, input.Invitee)
if err != nil {
return nil, err
}
if invitedSenderID != nil {
err = abortIfAlreadyJoined(ctx, input.RoomID, *invitedSenderID, input.MembershipQuerier)
if err != nil {
return nil, err
}
}
stateNeeded, err := StateNeededForProtoEvent(&input.EventTemplate)
if err != nil {
return nil, err
}
if len(stateNeeded.Tuples()) == 0 {
return nil, spec.InternalServerError{}
}
latestEvents, err := input.EventQuerier(ctx, input.RoomID, stateNeeded.Tuples())
if err != nil {
return nil, err
}
if !latestEvents.RoomExists {
return nil, spec.InternalServerError{}
}
input.EventTemplate.Depth = latestEvents.Depth
authEvents, _ := NewAuthEvents(nil)
for _, event := range latestEvents.StateEvents {
err := authEvents.AddEvent(event)
if err != nil {
return nil, fmt.Errorf("authEvents.AddEvent: %w", err)
}
}
refs, err := stateNeeded.AuthEventReferences(authEvents)
if err != nil {
return nil, fmt.Errorf("eventsNeeded.AuthEventReferences: %w", err)
}
input.EventTemplate.AuthEvents, input.EventTemplate.PrevEvents = truncateAuthAndPrevEvents(refs, latestEvents.PrevEventIDs)
checkEventAllowed := func(inviteEvent PDU) error {
// The invite originated locally. Therefore we have a responsibility to
// try and see if the user is allowed to make this invite. We can't do
// this for invites coming in over federation - we have to take those on
// trust.
authEventProvider, err := input.StateQuerier.GetAuthEvents(ctx, inviteEvent)
if err != nil {
logger.WithError(err).WithField("event_id", inviteEvent.EventID()).WithField("auth_event_ids", inviteEvent.AuthEventIDs()).Error(
"ProcessInvite.getAuthEvents failed for event",
)
return spec.Forbidden(err.Error())
}
// Check if the event is allowed.
if err = Allowed(inviteEvent, authEventProvider, input.UserIDQuerier); err != nil {
logger.WithError(err).WithField("event_id", inviteEvent.EventID()).WithField("auth_event_ids", inviteEvent.AuthEventIDs()).Error(
"ProcessInvite: event not allowed",
)
return spec.Forbidden(err.Error())
}
return nil
}
// If the target isn't local then we should send the invite
// over federation. It might be that the remote user doesn't exist,
// in which case we can give up processing here.
var inviteEvent PDU
switch input.RoomVersion {
case RoomVersionPseudoIDs:
keyID := KeyID("ed25519:1")
origin := spec.ServerName(spec.SenderIDFromPseudoIDKey(input.SigningKey))
if input.IsTargetLocal {
// if we invited a local user, we can also create a user room key, if it doesn't exist yet.
inviteeSenderID, inviteeSigningKey, err := input.SenderIDCreator(ctx, input.Invitee, input.RoomID, string(input.RoomVersion))
if err != nil {
return nil, err
}
inviteeSenderIDString := string(inviteeSenderID)
input.EventTemplate.StateKey = &inviteeSenderIDString
// Sign the event so that other servers will know that we have received the invite.
fullEventBuilder := verImpl.NewEventBuilderFromProtoEvent(&input.EventTemplate)
inviteEvent, err = fullEventBuilder.Build(input.EventTime, spec.ServerName(inviteeSenderID), keyID, inviteeSigningKey)
if err != nil {
logger.WithError(err).Error("failed building invite event")
return nil, spec.InternalServerError{}
}
// Have the inviter also sign the event
inviteEvent = inviteEvent.Sign(string(origin), keyID, input.SigningKey)
verifier := JSONVerifierSelf{}
err = VerifyEventSignatures(ctx, inviteEvent, verifier, input.UserIDQuerier)
if err != nil {
logger.WithError(err).Error("local invite event has invalid signatures")
return nil, spec.Forbidden(err.Error())
}
err = checkEventAllowed(inviteEvent)
if err != nil {
return nil, err
}
} else {
inviteEvent, err = fedClient.SendInviteV3(ctx, input.EventTemplate, input.Invitee, input.RoomVersion, inviteState)
if err != nil {
logger.WithError(err).Error("fedClient.SendInviteV3 failed")
return nil, spec.Forbidden(err.Error())
}
logger.Debugf("Federated SendInviteV3 success to user %s", input.Invitee.String())
// Have the inviter also sign the event
inviteEvent = inviteEvent.Sign(
string(origin), keyID, input.SigningKey,
)
verifier := JSONVerifierSelf{}
err := VerifyEventSignatures(ctx, inviteEvent, verifier, input.UserIDQuerier)
if err != nil {
logger.WithError(err).Error("fedClient.SendInviteV3 returned event with invalid signatures")
return nil, spec.Forbidden(err.Error())
}
err = input.StoreSenderIDFromPublicID(ctx, spec.SenderID(*inviteEvent.StateKey()), input.Invitee.String(), input.RoomID)
if err != nil {
logger.WithError(err).Errorf("failed storing senderID for %s", input.Invitee.String())
return nil, spec.InternalServerError{}
}
// TODO: This should happen before the federation call ideally,
// but we don't have a full PDU yet in this case by that point.
err = checkEventAllowed(inviteEvent)
if err != nil {
return nil, err
}
}
default:
inviteeSenderID := input.Invitee.String()
input.EventTemplate.StateKey = &inviteeSenderID
// Sign the event so that other servers will know that we have received the invite.
fullEventBuilder := verImpl.NewEventBuilderFromProtoEvent(&input.EventTemplate)
fullEvent, err := fullEventBuilder.Build(input.EventTime, input.Inviter.Domain(), input.KeyID, input.SigningKey)
if err != nil {
logger.WithError(err).Error("failed building invite event")
return nil, spec.InternalServerError{}
}
inviteEvent = fullEvent.Sign(
string(input.Invitee.Domain()), input.KeyID, input.SigningKey,
)
err = checkEventAllowed(inviteEvent)
if err != nil {
return nil, err
}
if !input.IsTargetLocal {
eventID := inviteEvent.EventID()
inviteEvent, err = fedClient.SendInvite(ctx, inviteEvent, inviteState)
if err != nil {
logger.WithError(err).WithField("event_id", eventID).Error("fedClient.SendInvite failed")
return nil, spec.Forbidden(err.Error())
}
logger.Debugf("Federated SendInvite success with event ID %s", eventID)
}
}
return inviteEvent, nil
}
// truncateAuthAndPrevEvents limits the number of events we add into
// an event as prev_events or auth_events.
// NOTSPEC: The limits here feel a bit arbitrary but they are currently
// here because of https://github.com/matrix-org/matrix-doc/issues/2307
// and because Synapse will just drop events that don't comply.
func truncateAuthAndPrevEvents(auth, prev []string) (
truncAuth, truncPrev []string,
) {
truncAuth, truncPrev = auth, prev
if len(truncAuth) > 10 {
truncAuth = truncAuth[:10]
}
if len(truncPrev) > 20 {
truncPrev = truncPrev[:20]
}
return
}