Host /unstable and /r0 versions of r0 APIs
							parent
							
								
									af96c6f4d3
								
							
						
					
					
						commit
						14d7acfad4
					
				|  | @ -165,7 +165,7 @@ class BaseFederationServlet(object): | |||
|             if code is None: | ||||
|                 continue | ||||
| 
 | ||||
|             server.register_path(method, pattern, self._wrap(code)) | ||||
|             server.register_paths(method, (pattern,), self._wrap(code)) | ||||
| 
 | ||||
| 
 | ||||
| class FederationSendServlet(BaseFederationServlet): | ||||
|  |  | |||
|  | @ -120,7 +120,7 @@ class HttpServer(object): | |||
|     """ Interface for registering callbacks on a HTTP server | ||||
|     """ | ||||
| 
 | ||||
|     def register_path(self, method, path_pattern, callback): | ||||
|     def register_paths(self, method, path_patterns, callback): | ||||
|         """ Register a callback that gets fired if we receive a http request | ||||
|         with the given method for a path that matches the given regex. | ||||
| 
 | ||||
|  | @ -129,7 +129,7 @@ class HttpServer(object): | |||
| 
 | ||||
|         Args: | ||||
|             method (str): The method to listen to. | ||||
|             path_pattern (str): The regex used to match requests. | ||||
|             path_patterns (list<SRE_Pattern>): The regex used to match requests. | ||||
|             callback (function): The function to fire if we receive a matched | ||||
|                 request. The first argument will be the request object and | ||||
|                 subsequent arguments will be any matched groups from the regex. | ||||
|  | @ -165,10 +165,11 @@ class JsonResource(HttpServer, resource.Resource): | |||
|         self.version_string = hs.version_string | ||||
|         self.hs = hs | ||||
| 
 | ||||
|     def register_path(self, method, path_pattern, callback): | ||||
|         self.path_regexs.setdefault(method, []).append( | ||||
|             self._PathEntry(path_pattern, callback) | ||||
|         ) | ||||
|     def register_paths(self, method, path_patterns, callback): | ||||
|         for path_pattern in path_patterns: | ||||
|             self.path_regexs.setdefault(method, []).append( | ||||
|                 self._PathEntry(path_pattern, callback) | ||||
|             ) | ||||
| 
 | ||||
|     def render(self, request): | ||||
|         """ This gets called by twisted every time someone sends us a request. | ||||
|  |  | |||
|  | @ -19,7 +19,6 @@ from synapse.api.errors import SynapseError | |||
| 
 | ||||
| import logging | ||||
| 
 | ||||
| 
 | ||||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| 
 | ||||
|  | @ -102,12 +101,13 @@ class RestServlet(object): | |||
| 
 | ||||
|     def register(self, http_server): | ||||
|         """ Register this servlet with the given HTTP server. """ | ||||
|         if hasattr(self, "PATTERN"): | ||||
|             pattern = self.PATTERN | ||||
|         if hasattr(self, "PATTERNS"): | ||||
|             patterns = self.PATTERNS | ||||
| 
 | ||||
|             for method in ("GET", "PUT", "POST", "OPTIONS", "DELETE"): | ||||
|                 if hasattr(self, "on_%s" % (method,)): | ||||
|                     method_handler = getattr(self, "on_%s" % (method,)) | ||||
|                     http_server.register_path(method, pattern, method_handler) | ||||
|                     http_server.register_paths(method, patterns, method_handler) | ||||
| 
 | ||||
|         else: | ||||
|             raise NotImplementedError("RestServlet must register something.") | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| from synapse.api.errors import AuthError, SynapseError | ||||
| from synapse.types import UserID | ||||
| 
 | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| import logging | ||||
| 
 | ||||
|  | @ -26,7 +26,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class WhoisRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/admin/whois/(?P<user_id>[^/]*)") | ||||
|     PATTERNS = client_path_patterns("/admin/whois/(?P<user_id>[^/]*)", releases=()) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  |  | |||
|  | @ -27,7 +27,7 @@ import logging | |||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| 
 | ||||
| def client_path_pattern(path_regex): | ||||
| def client_path_patterns(path_regex, releases=(0,)): | ||||
|     """Creates a regex compiled client path with the correct client path | ||||
|     prefix. | ||||
| 
 | ||||
|  | @ -37,7 +37,13 @@ def client_path_pattern(path_regex): | |||
|     Returns: | ||||
|         SRE_Pattern | ||||
|     """ | ||||
|     return re.compile("^" + CLIENT_PREFIX + path_regex) | ||||
|     patterns = [re.compile("^" + CLIENT_PREFIX + path_regex)] | ||||
|     unstable_prefix = CLIENT_PREFIX.replace("/api/v1", "/unstable") | ||||
|     patterns.append(re.compile("^" + unstable_prefix + path_regex)) | ||||
|     for release in releases: | ||||
|         new_prefix = CLIENT_PREFIX.replace("/api/v1", "/r%d" % release) | ||||
|         patterns.append(re.compile("^" + new_prefix + path_regex)) | ||||
|     return patterns | ||||
| 
 | ||||
| 
 | ||||
| class ClientV1RestServlet(RestServlet): | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import AuthError, SynapseError, Codes | ||||
| from synapse.types import RoomAlias | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| import logging | ||||
|  | @ -32,7 +32,7 @@ def register_servlets(hs, http_server): | |||
| 
 | ||||
| 
 | ||||
| class ClientDirectoryServer(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/directory/room/(?P<room_alias>[^/]*)$") | ||||
|     PATTERNS = client_path_patterns("/directory/room/(?P<room_alias>[^/]*)$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, room_alias): | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import SynapseError | ||||
| from synapse.streams.config import PaginationConfig | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| from synapse.events.utils import serialize_event | ||||
| 
 | ||||
| import logging | ||||
|  | @ -28,7 +28,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class EventStreamRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/events$") | ||||
|     PATTERNS = client_path_patterns("/events$") | ||||
| 
 | ||||
|     DEFAULT_LONGPOLL_TIME_MS = 30000 | ||||
| 
 | ||||
|  | @ -72,7 +72,7 @@ class EventStreamRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Unit test gets, with and without auth, with different kinds of events. | ||||
| class EventRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/events/(?P<event_id>[^/]*)$") | ||||
|     PATTERNS = client_path_patterns("/events/(?P<event_id>[^/]*)$") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(EventRestServlet, self).__init__(hs) | ||||
|  |  | |||
|  | @ -16,12 +16,12 @@ | |||
| from twisted.internet import defer | ||||
| 
 | ||||
| from synapse.streams.config import PaginationConfig | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| 
 | ||||
| # TODO: Needs unit testing | ||||
| class InitialSyncRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/initialSync$") | ||||
|     PATTERNS = client_path_patterns("/initialSync$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request): | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| from synapse.api.errors import SynapseError, LoginError, Codes | ||||
| from synapse.http.client import SimpleHttpClient | ||||
| from synapse.types import UserID | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| import urllib | ||||
|  | @ -36,7 +36,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class LoginRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/login$") | ||||
|     PATTERNS = client_path_patterns("/login$", releases=()) | ||||
|     PASS_TYPE = "m.login.password" | ||||
|     SAML2_TYPE = "m.login.saml2" | ||||
|     CAS_TYPE = "m.login.cas" | ||||
|  | @ -238,7 +238,7 @@ class LoginRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class SAML2RestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/login/saml2") | ||||
|     PATTERNS = client_path_patterns("/login/saml2", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(SAML2RestServlet, self).__init__(hs) | ||||
|  | @ -282,7 +282,7 @@ class SAML2RestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO Delete this after all CAS clients switch to token login instead | ||||
| class CasRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/login/cas") | ||||
|     PATTERNS = client_path_patterns("/login/cas", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(CasRestServlet, self).__init__(hs) | ||||
|  | @ -293,7 +293,7 @@ class CasRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class CasRedirectServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/login/cas/redirect") | ||||
|     PATTERNS = client_path_patterns("/login/cas/redirect", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(CasRedirectServlet, self).__init__(hs) | ||||
|  | @ -316,7 +316,7 @@ class CasRedirectServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class CasTicketServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/login/cas/ticket") | ||||
|     PATTERNS = client_path_patterns("/login/cas/ticket", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(CasTicketServlet, self).__init__(hs) | ||||
|  |  | |||
|  | @ -19,7 +19,7 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import SynapseError | ||||
| from synapse.types import UserID | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| import logging | ||||
|  | @ -28,7 +28,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class PresenceStatusRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/presence/(?P<user_id>[^/]*)/status") | ||||
|     PATTERNS = client_path_patterns("/presence/(?P<user_id>[^/]*)/status") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  | @ -73,7 +73,7 @@ class PresenceStatusRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class PresenceListRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/presence/list/(?P<user_id>[^/]*)") | ||||
|     PATTERNS = client_path_patterns("/presence/list/(?P<user_id>[^/]*)") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  |  | |||
|  | @ -16,14 +16,14 @@ | |||
| """ This module contains REST servlets to do with profile: /profile/<paths> """ | ||||
| from twisted.internet import defer | ||||
| 
 | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| from synapse.types import UserID | ||||
| 
 | ||||
| import simplejson as json | ||||
| 
 | ||||
| 
 | ||||
| class ProfileDisplaynameRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/profile/(?P<user_id>[^/]*)/displayname") | ||||
|     PATTERNS = client_path_patterns("/profile/(?P<user_id>[^/]*)/displayname") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  | @ -56,7 +56,7 @@ class ProfileDisplaynameRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class ProfileAvatarURLRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/profile/(?P<user_id>[^/]*)/avatar_url") | ||||
|     PATTERNS = client_path_patterns("/profile/(?P<user_id>[^/]*)/avatar_url") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  | @ -89,7 +89,7 @@ class ProfileAvatarURLRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class ProfileRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/profile/(?P<user_id>[^/]*)") | ||||
|     PATTERNS = client_path_patterns("/profile/(?P<user_id>[^/]*)") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, user_id): | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| from synapse.api.errors import ( | ||||
|     SynapseError, Codes, UnrecognizedRequestError, NotFoundError, StoreError | ||||
| ) | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| from synapse.storage.push_rule import ( | ||||
|     InconsistentRuleException, RuleNotFoundException | ||||
| ) | ||||
|  | @ -31,7 +31,7 @@ import simplejson as json | |||
| 
 | ||||
| 
 | ||||
| class PushRuleRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/pushrules/.*$") | ||||
|     PATTERNS = client_path_patterns("/pushrules/.*$") | ||||
|     SLIGHTLY_PEDANTIC_TRAILING_SLASH_ERROR = ( | ||||
|         "Unrecognised request: You probably wanted a trailing slash") | ||||
| 
 | ||||
|  |  | |||
|  | @ -17,13 +17,13 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import SynapseError, Codes | ||||
| from synapse.push import PusherConfigException | ||||
| from .base import ClientV1RestServlet, client_path_pattern | ||||
| from .base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| 
 | ||||
| 
 | ||||
| class PusherRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/pushers/set$") | ||||
|     PATTERNS = client_path_patterns("/pushers/set$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_POST(self, request): | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import SynapseError, Codes | ||||
| from synapse.api.constants import LoginType | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| import synapse.util.stringutils as stringutils | ||||
| 
 | ||||
| from synapse.util.async import run_on_reactor | ||||
|  | @ -48,7 +48,7 @@ class RegisterRestServlet(ClientV1RestServlet): | |||
|     handler doesn't have a concept of multi-stages or sessions. | ||||
|     """ | ||||
| 
 | ||||
|     PATTERN = client_path_pattern("/register$") | ||||
|     PATTERNS = client_path_patterns("/register$", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(RegisterRestServlet, self).__init__(hs) | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
| """ This module contains REST servlets to do with rooms: /rooms/<paths> """ | ||||
| from twisted.internet import defer | ||||
| 
 | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| from synapse.api.errors import SynapseError, Codes, AuthError | ||||
| from synapse.streams.config import PaginationConfig | ||||
| from synapse.api.constants import EventTypes, Membership | ||||
|  | @ -34,16 +34,16 @@ class RoomCreateRestServlet(ClientV1RestServlet): | |||
|     # No PATTERN; we have custom dispatch rules here | ||||
| 
 | ||||
|     def register(self, http_server): | ||||
|         PATTERN = "/createRoom" | ||||
|         register_txn_path(self, PATTERN, http_server) | ||||
|         PATTERNS = "/createRoom" | ||||
|         register_txn_path(self, PATTERNS, http_server) | ||||
|         # define CORS for all of /rooms in RoomCreateRestServlet for simplicity | ||||
|         http_server.register_path("OPTIONS", | ||||
|                                   client_path_pattern("/rooms(?:/.*)?$"), | ||||
|                                   self.on_OPTIONS) | ||||
|         http_server.register_paths("OPTIONS", | ||||
|                                    client_path_patterns("/rooms(?:/.*)?$"), | ||||
|                                    self.on_OPTIONS) | ||||
|         # define CORS for /createRoom[/txnid] | ||||
|         http_server.register_path("OPTIONS", | ||||
|                                   client_path_pattern("/createRoom(?:/.*)?$"), | ||||
|                                   self.on_OPTIONS) | ||||
|         http_server.register_paths("OPTIONS", | ||||
|                                    client_path_patterns("/createRoom(?:/.*)?$"), | ||||
|                                    self.on_OPTIONS) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_PUT(self, request, txn_id): | ||||
|  | @ -103,18 +103,18 @@ class RoomStateEventRestServlet(ClientV1RestServlet): | |||
|         state_key = ("/rooms/(?P<room_id>[^/]*)/state/" | ||||
|                      "(?P<event_type>[^/]*)/(?P<state_key>[^/]*)$") | ||||
| 
 | ||||
|         http_server.register_path("GET", | ||||
|                                   client_path_pattern(state_key), | ||||
|                                   self.on_GET) | ||||
|         http_server.register_path("PUT", | ||||
|                                   client_path_pattern(state_key), | ||||
|                                   self.on_PUT) | ||||
|         http_server.register_path("GET", | ||||
|                                   client_path_pattern(no_state_key), | ||||
|                                   self.on_GET_no_state_key) | ||||
|         http_server.register_path("PUT", | ||||
|                                   client_path_pattern(no_state_key), | ||||
|                                   self.on_PUT_no_state_key) | ||||
|         http_server.register_paths("GET", | ||||
|                                    client_path_patterns(state_key), | ||||
|                                    self.on_GET) | ||||
|         http_server.register_paths("PUT", | ||||
|                                    client_path_patterns(state_key), | ||||
|                                    self.on_PUT) | ||||
|         http_server.register_paths("GET", | ||||
|                                    client_path_patterns(no_state_key, releases=()), | ||||
|                                    self.on_GET_no_state_key) | ||||
|         http_server.register_paths("PUT", | ||||
|                                    client_path_patterns(no_state_key, releases=()), | ||||
|                                    self.on_PUT_no_state_key) | ||||
| 
 | ||||
|     def on_GET_no_state_key(self, request, room_id, event_type): | ||||
|         return self.on_GET(request, room_id, event_type, "") | ||||
|  | @ -170,8 +170,8 @@ class RoomSendEventRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
|     def register(self, http_server): | ||||
|         # /rooms/$roomid/send/$event_type[/$txn_id] | ||||
|         PATTERN = ("/rooms/(?P<room_id>[^/]*)/send/(?P<event_type>[^/]*)") | ||||
|         register_txn_path(self, PATTERN, http_server, with_get=True) | ||||
|         PATTERNS = ("/rooms/(?P<room_id>[^/]*)/send/(?P<event_type>[^/]*)") | ||||
|         register_txn_path(self, PATTERNS, http_server, with_get=True) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_POST(self, request, room_id, event_type, txn_id=None): | ||||
|  | @ -215,8 +215,8 @@ class JoinRoomAliasServlet(ClientV1RestServlet): | |||
| 
 | ||||
|     def register(self, http_server): | ||||
|         # /join/$room_identifier[/$txn_id] | ||||
|         PATTERN = ("/join/(?P<room_identifier>[^/]*)") | ||||
|         register_txn_path(self, PATTERN, http_server) | ||||
|         PATTERNS = ("/join/(?P<room_identifier>[^/]*)") | ||||
|         register_txn_path(self, PATTERNS, http_server) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_POST(self, request, room_identifier, txn_id=None): | ||||
|  | @ -280,7 +280,7 @@ class JoinRoomAliasServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Needs unit testing | ||||
| class PublicRoomListRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/publicRooms$") | ||||
|     PATTERNS = client_path_patterns("/publicRooms$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request): | ||||
|  | @ -291,7 +291,7 @@ class PublicRoomListRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Needs unit testing | ||||
| class RoomMemberListRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/rooms/(?P<room_id>[^/]*)/members$") | ||||
|     PATTERNS = client_path_patterns("/rooms/(?P<room_id>[^/]*)/members$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, room_id): | ||||
|  | @ -328,7 +328,7 @@ class RoomMemberListRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Needs better unit testing | ||||
| class RoomMessageListRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/rooms/(?P<room_id>[^/]*)/messages$") | ||||
|     PATTERNS = client_path_patterns("/rooms/(?P<room_id>[^/]*)/messages$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, room_id): | ||||
|  | @ -351,7 +351,7 @@ class RoomMessageListRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Needs unit testing | ||||
| class RoomStateRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/rooms/(?P<room_id>[^/]*)/state$") | ||||
|     PATTERNS = client_path_patterns("/rooms/(?P<room_id>[^/]*)/state$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, room_id): | ||||
|  | @ -368,7 +368,7 @@ class RoomStateRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| # TODO: Needs unit testing | ||||
| class RoomInitialSyncRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/rooms/(?P<room_id>[^/]*)/initialSync$") | ||||
|     PATTERNS = client_path_patterns("/rooms/(?P<room_id>[^/]*)/initialSync$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request, room_id): | ||||
|  | @ -384,7 +384,7 @@ class RoomInitialSyncRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class RoomTriggerBackfill(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/rooms/(?P<room_id>[^/]*)/backfill$") | ||||
|     PATTERNS = client_path_patterns("/rooms/(?P<room_id>[^/]*)/backfill$", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(RoomTriggerBackfill, self).__init__(hs) | ||||
|  | @ -408,7 +408,7 @@ class RoomTriggerBackfill(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class RoomEventContext(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern( | ||||
|     PATTERNS = client_path_patterns( | ||||
|         "/rooms/(?P<room_id>[^/]*)/context/(?P<event_id>[^/]*)$" | ||||
|     ) | ||||
| 
 | ||||
|  | @ -447,9 +447,9 @@ class RoomMembershipRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
|     def register(self, http_server): | ||||
|         # /rooms/$roomid/[invite|join|leave] | ||||
|         PATTERN = ("/rooms/(?P<room_id>[^/]*)/" | ||||
|                    "(?P<membership_action>join|invite|leave|ban|kick|forget)") | ||||
|         register_txn_path(self, PATTERN, http_server) | ||||
|         PATTERNS = ("/rooms/(?P<room_id>[^/]*)/" | ||||
|                     "(?P<membership_action>join|invite|leave|ban|kick|forget)") | ||||
|         register_txn_path(self, PATTERNS, http_server) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_POST(self, request, room_id, membership_action, txn_id=None): | ||||
|  | @ -543,8 +543,8 @@ class RoomMembershipRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| class RoomRedactEventRestServlet(ClientV1RestServlet): | ||||
|     def register(self, http_server): | ||||
|         PATTERN = ("/rooms/(?P<room_id>[^/]*)/redact/(?P<event_id>[^/]*)") | ||||
|         register_txn_path(self, PATTERN, http_server) | ||||
|         PATTERNS = ("/rooms/(?P<room_id>[^/]*)/redact/(?P<event_id>[^/]*)") | ||||
|         register_txn_path(self, PATTERNS, http_server) | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_POST(self, request, room_id, event_id, txn_id=None): | ||||
|  | @ -582,7 +582,7 @@ class RoomRedactEventRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class RoomTypingRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern( | ||||
|     PATTERNS = client_path_patterns( | ||||
|         "/rooms/(?P<room_id>[^/]*)/typing/(?P<user_id>[^/]*)$" | ||||
|     ) | ||||
| 
 | ||||
|  | @ -615,7 +615,7 @@ class RoomTypingRestServlet(ClientV1RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class SearchRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern( | ||||
|     PATTERNS = client_path_patterns( | ||||
|         "/search$" | ||||
|     ) | ||||
| 
 | ||||
|  | @ -655,20 +655,20 @@ def register_txn_path(servlet, regex_string, http_server, with_get=False): | |||
|         http_server : The http_server to register paths with. | ||||
|         with_get: True to also register respective GET paths for the PUTs. | ||||
|     """ | ||||
|     http_server.register_path( | ||||
|     http_server.register_paths( | ||||
|         "POST", | ||||
|         client_path_pattern(regex_string + "$"), | ||||
|         client_path_patterns(regex_string + "$"), | ||||
|         servlet.on_POST | ||||
|     ) | ||||
|     http_server.register_path( | ||||
|     http_server.register_paths( | ||||
|         "PUT", | ||||
|         client_path_pattern(regex_string + "/(?P<txn_id>[^/]*)$"), | ||||
|         client_path_patterns(regex_string + "/(?P<txn_id>[^/]*)$"), | ||||
|         servlet.on_PUT | ||||
|     ) | ||||
|     if with_get: | ||||
|         http_server.register_path( | ||||
|         http_server.register_paths( | ||||
|             "GET", | ||||
|             client_path_pattern(regex_string + "/(?P<txn_id>[^/]*)$"), | ||||
|             client_path_patterns(regex_string + "/(?P<txn_id>[^/]*)$"), | ||||
|             servlet.on_GET | ||||
|         ) | ||||
| 
 | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
| 
 | ||||
| from twisted.internet import defer | ||||
| 
 | ||||
| from base import ClientV1RestServlet, client_path_pattern | ||||
| from base import ClientV1RestServlet, client_path_patterns | ||||
| 
 | ||||
| 
 | ||||
| import hmac | ||||
|  | @ -24,7 +24,7 @@ import base64 | |||
| 
 | ||||
| 
 | ||||
| class VoipRestServlet(ClientV1RestServlet): | ||||
|     PATTERN = client_path_pattern("/voip/turnServer$") | ||||
|     PATTERNS = client_path_patterns("/voip/turnServer$") | ||||
| 
 | ||||
|     @defer.inlineCallbacks | ||||
|     def on_GET(self, request): | ||||
|  |  | |||
|  | @ -27,7 +27,7 @@ import simplejson | |||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| 
 | ||||
| def client_v2_pattern(path_regex): | ||||
| def client_v2_patterns(path_regex, releases=(0,)): | ||||
|     """Creates a regex compiled client path with the correct client path | ||||
|     prefix. | ||||
| 
 | ||||
|  | @ -37,7 +37,13 @@ def client_v2_pattern(path_regex): | |||
|     Returns: | ||||
|         SRE_Pattern | ||||
|     """ | ||||
|     return re.compile("^" + CLIENT_V2_ALPHA_PREFIX + path_regex) | ||||
|     patterns = [re.compile("^" + CLIENT_V2_ALPHA_PREFIX + path_regex)] | ||||
|     unstable_prefix = CLIENT_V2_ALPHA_PREFIX.replace("/v2_alpha", "/unstable") | ||||
|     patterns.append(re.compile("^" + unstable_prefix + path_regex)) | ||||
|     for release in releases: | ||||
|         new_prefix = CLIENT_V2_ALPHA_PREFIX.replace("/v2_alpha", "/r%d" % release) | ||||
|         patterns.append(re.compile("^" + new_prefix + path_regex)) | ||||
|     return patterns | ||||
| 
 | ||||
| 
 | ||||
| def parse_request_allow_empty(request): | ||||
|  |  | |||
|  | @ -20,7 +20,7 @@ from synapse.api.errors import LoginError, SynapseError, Codes | |||
| from synapse.http.servlet import RestServlet | ||||
| from synapse.util.async import run_on_reactor | ||||
| 
 | ||||
| from ._base import client_v2_pattern, parse_json_dict_from_request | ||||
| from ._base import client_v2_patterns, parse_json_dict_from_request | ||||
| 
 | ||||
| import logging | ||||
| 
 | ||||
|  | @ -29,7 +29,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class PasswordRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern("/account/password") | ||||
|     PATTERNS = client_v2_patterns("/account/password", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(PasswordRestServlet, self).__init__() | ||||
|  | @ -89,7 +89,7 @@ class PasswordRestServlet(RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class ThreepidRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern("/account/3pid") | ||||
|     PATTERNS = client_v2_patterns("/account/3pid", releases=()) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(ThreepidRestServlet, self).__init__() | ||||
|  |  | |||
|  | @ -20,7 +20,7 @@ from synapse.api.errors import SynapseError | |||
| from synapse.api.urls import CLIENT_V2_ALPHA_PREFIX | ||||
| from synapse.http.servlet import RestServlet | ||||
| 
 | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| import logging | ||||
| 
 | ||||
|  | @ -97,7 +97,7 @@ class AuthRestServlet(RestServlet): | |||
|     cannot be handled in the normal flow (with requests to the same endpoint). | ||||
|     Current use is for web fallback auth. | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern("/auth/(?P<stagetype>[\w\.]*)/fallback/web") | ||||
|     PATTERNS = client_v2_patterns("/auth/(?P<stagetype>[\w\.]*)/fallback/web") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(AuthRestServlet, self).__init__() | ||||
|  |  | |||
|  | @ -19,7 +19,7 @@ from synapse.api.errors import AuthError, SynapseError | |||
| from synapse.http.servlet import RestServlet | ||||
| from synapse.types import UserID | ||||
| 
 | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| import logging | ||||
|  | @ -29,7 +29,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class GetFilterRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern("/user/(?P<user_id>[^/]*)/filter/(?P<filter_id>[^/]*)") | ||||
|     PATTERNS = client_v2_patterns("/user/(?P<user_id>[^/]*)/filter/(?P<filter_id>[^/]*)") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(GetFilterRestServlet, self).__init__() | ||||
|  | @ -65,7 +65,7 @@ class GetFilterRestServlet(RestServlet): | |||
| 
 | ||||
| 
 | ||||
| class CreateFilterRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern("/user/(?P<user_id>[^/]*)/filter") | ||||
|     PATTERNS = client_v2_patterns("/user/(?P<user_id>[^/]*)/filter") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(CreateFilterRestServlet, self).__init__() | ||||
|  |  | |||
|  | @ -21,7 +21,7 @@ from synapse.types import UserID | |||
| 
 | ||||
| from canonicaljson import encode_canonical_json | ||||
| 
 | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| import simplejson as json | ||||
| import logging | ||||
|  | @ -54,7 +54,7 @@ class KeyUploadServlet(RestServlet): | |||
|       }, | ||||
|     } | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern("/keys/upload/(?P<device_id>[^/]*)") | ||||
|     PATTERNS = client_v2_patterns("/keys/upload/(?P<device_id>[^/]*)") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(KeyUploadServlet, self).__init__() | ||||
|  | @ -154,12 +154,13 @@ class KeyQueryServlet(RestServlet): | |||
|     } } } } } } | ||||
|     """ | ||||
| 
 | ||||
|     PATTERN = client_v2_pattern( | ||||
|     PATTERNS = client_v2_patterns( | ||||
|         "/keys/query(?:" | ||||
|         "/(?P<user_id>[^/]*)(?:" | ||||
|         "/(?P<device_id>[^/]*)" | ||||
|         ")?" | ||||
|         ")?" | ||||
|         ")?", | ||||
|         releases=() | ||||
|     ) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|  | @ -245,10 +246,11 @@ class OneTimeKeyServlet(RestServlet): | |||
|     } } } } | ||||
| 
 | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern( | ||||
|     PATTERNS = client_v2_patterns( | ||||
|         "/keys/claim(?:/?|(?:/" | ||||
|         "(?P<user_id>[^/]*)/(?P<device_id>[^/]*)/(?P<algorithm>[^/]*)" | ||||
|         ")?)" | ||||
|         ")?)", | ||||
|         releases=() | ||||
|     ) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|  |  | |||
|  | @ -17,7 +17,7 @@ from twisted.internet import defer | |||
| 
 | ||||
| from synapse.api.errors import SynapseError | ||||
| from synapse.http.servlet import RestServlet | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| import logging | ||||
| 
 | ||||
|  | @ -26,7 +26,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class ReceiptRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern( | ||||
|     PATTERNS = client_v2_patterns( | ||||
|         "/rooms/(?P<room_id>[^/]*)" | ||||
|         "/receipt/(?P<receipt_type>[^/]*)" | ||||
|         "/(?P<event_id>[^/]*)$" | ||||
|  |  | |||
|  | @ -19,7 +19,7 @@ from synapse.api.constants import LoginType | |||
| from synapse.api.errors import SynapseError, Codes, UnrecognizedRequestError | ||||
| from synapse.http.servlet import RestServlet | ||||
| 
 | ||||
| from ._base import client_v2_pattern, parse_json_dict_from_request | ||||
| from ._base import client_v2_patterns, parse_json_dict_from_request | ||||
| 
 | ||||
| import logging | ||||
| import hmac | ||||
|  | @ -41,7 +41,7 @@ logger = logging.getLogger(__name__) | |||
| 
 | ||||
| 
 | ||||
| class RegisterRestServlet(RestServlet): | ||||
|     PATTERN = client_v2_pattern("/register") | ||||
|     PATTERNS = client_v2_patterns("/register") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(RegisterRestServlet, self).__init__() | ||||
|  |  | |||
|  | @ -25,7 +25,7 @@ from synapse.events.utils import ( | |||
|     serialize_event, format_event_for_client_v2_without_room_id, | ||||
| ) | ||||
| from synapse.api.filtering import FilterCollection | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| import copy | ||||
| import logging | ||||
|  | @ -69,7 +69,7 @@ class SyncRestServlet(RestServlet): | |||
|         } | ||||
|     """ | ||||
| 
 | ||||
|     PATTERN = client_v2_pattern("/sync$") | ||||
|     PATTERNS = client_v2_patterns("/sync$") | ||||
|     ALLOWED_PRESENCE = set(["online", "offline"]) | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ | |||
| # See the License for the specific language governing permissions and | ||||
| # limitations under the License. | ||||
| 
 | ||||
| from ._base import client_v2_pattern | ||||
| from ._base import client_v2_patterns | ||||
| 
 | ||||
| from synapse.http.servlet import RestServlet | ||||
| from synapse.api.errors import AuthError, SynapseError | ||||
|  | @ -31,7 +31,7 @@ class TagListServlet(RestServlet): | |||
|     """ | ||||
|     GET /user/{user_id}/rooms/{room_id}/tags HTTP/1.1 | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern( | ||||
|     PATTERNS = client_v2_patterns( | ||||
|         "/user/(?P<user_id>[^/]*)/rooms/(?P<room_id>[^/]*)/tags" | ||||
|     ) | ||||
| 
 | ||||
|  | @ -56,7 +56,7 @@ class TagServlet(RestServlet): | |||
|     PUT /user/{user_id}/rooms/{room_id}/tags/{tag} HTTP/1.1 | ||||
|     DELETE /user/{user_id}/rooms/{room_id}/tags/{tag} HTTP/1.1 | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern( | ||||
|     PATTERNS = client_v2_patterns( | ||||
|         "/user/(?P<user_id>[^/]*)/rooms/(?P<room_id>[^/]*)/tags/(?P<tag>[^/]*)" | ||||
|     ) | ||||
| 
 | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ from twisted.internet import defer | |||
| from synapse.api.errors import AuthError, StoreError, SynapseError | ||||
| from synapse.http.servlet import RestServlet | ||||
| 
 | ||||
| from ._base import client_v2_pattern, parse_json_dict_from_request | ||||
| from ._base import client_v2_patterns, parse_json_dict_from_request | ||||
| 
 | ||||
| 
 | ||||
| class TokenRefreshRestServlet(RestServlet): | ||||
|  | @ -26,7 +26,7 @@ class TokenRefreshRestServlet(RestServlet): | |||
|     Exchanges refresh tokens for a pair of an access token and a new refresh | ||||
|     token. | ||||
|     """ | ||||
|     PATTERN = client_v2_pattern("/tokenrefresh") | ||||
|     PATTERNS = client_v2_patterns("/tokenrefresh") | ||||
| 
 | ||||
|     def __init__(self, hs): | ||||
|         super(TokenRefreshRestServlet, self).__init__() | ||||
|  |  | |||
|  | @ -168,8 +168,9 @@ class MockHttpResource(HttpServer): | |||
| 
 | ||||
|         raise KeyError("No event can handle %s" % path) | ||||
| 
 | ||||
|     def register_path(self, method, path_pattern, callback): | ||||
|         self.callbacks.append((method, path_pattern, callback)) | ||||
|     def register_paths(self, method, path_patterns, callback): | ||||
|         for path_pattern in path_patterns: | ||||
|             self.callbacks.append((method, path_pattern, callback)) | ||||
| 
 | ||||
| 
 | ||||
| class MockKey(object): | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	 Daniel Wagner-Hall
						Daniel Wagner-Hall