diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000000000000000000000000000000000000..723ef36f4e4f32c4560383aa5987c575a30c6535
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1 @@
+.idea
\ No newline at end of file
diff --git a/.gitmodules b/.gitmodules
new file mode 100644
index 0000000000000000000000000000000000000000..79c7c1ea054bf3ed90bc0be427d1350ee0519405
--- /dev/null
+++ b/.gitmodules
@@ -0,0 +1,3 @@
+[submodule "perxis-proto"]
+	path = perxis-proto
+	url = https://git.perx.ru/perxis/perxis-proto.git
diff --git a/Makefile b/Makefile
new file mode 100644
index 0000000000000000000000000000000000000000..d591223784f193f941f3975ba970047cb81710f3
--- /dev/null
+++ b/Makefile
@@ -0,0 +1,26 @@
+PROTODIR=perxis-proto/proto
+DSTDIR=clients
+ALLPROTO?=$(shell find $(PROTODIR) -name '*.proto' )
+# Убираем status.proto нужен только для front
+PROTOFILES=	$(filter-out $(PROTODIR)/status/status.proto, $(ALLPROTO))
+PROTOPYFILES=$(PROTOFILES:.proto=_pb2.py)
+PROTOPYGRPCFILES=$(PROTOFILES:.proto=_pb2_grpc.py)
+
+# Генерация grpc-клиентов для Python
+proto: grpcio-tools-check $(PROTOPYFILES) $(PROTOPYGRPCFILES)
+	@echo "Generated all protobuf Python files"
+
+%_pb2.py %_pb2_grpc.py: %.proto
+	@python3 -m grpc_tools.protoc -I$(PROTODIR) --experimental_allow_proto3_optional --python_out=$(DSTDIR) --grpc_python_out=$(DSTDIR) "$<"
+
+proto-clean:
+	@rm -f $(PROTOPYFILES) -f $(PROTOPYGRPCFILES)
+
+# Helpers
+#
+# Проверка, что установлен пакет grpcio-tools для генерации кода Python
+grpcio-tools-check:
+ifeq (,$(shell pip3 list | grep -F grpcio-tools))
+	$(error "grpcio-tools not found. Run \"pip3 install grpcio-tools\" \
+	or visit \"https://grpc.io/docs/languages/python/basics/#generating-client-and-server-code\" for more.\n")
+endif
diff --git a/README.md b/README.md
index 4d19cf6ecac1ab18241c9a000b0bd20522355eda..d75ed63dfd60e25803cae4d9c6b94a0ec5f0840f 100644
--- a/README.md
+++ b/README.md
@@ -1,2 +1,2 @@
-# perxis-py
+# Perxis Python SDK
 
diff --git a/clients/clients/clients_pb2.py b/clients/clients/clients_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..2e32ad7e924cc14e6d488bfd0fbb649b941bbf09
--- /dev/null
+++ b/clients/clients/clients_pb2.py
@@ -0,0 +1,182 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: clients/clients.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x15\x63lients/clients.proto\x12\x0f\x63ontent.clients\x1a\x1bgoogle/protobuf/empty.proto\"\x9d\x03\n\x06\x43lient\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x0c\n\x04name\x18\x03 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x06 \x01(\t\x12\x15\n\x08\x64isabled\x18\x07 \x01(\x08H\x00\x88\x01\x01\x12\x0f\n\x07role_id\x18\x08 \x01(\t\x12,\n\x05oauth\x18\n \x01(\x0b\x32\x1d.content.clients.Client.OAuth\x12(\n\x03tls\x18\x0b \x01(\x0b\x32\x1b.content.clients.Client.TLS\x12/\n\x07\x61pi_key\x18\x0c \x01(\x0b\x32\x1e.content.clients.Client.APIKey\x1aU\n\x05OAuth\x12\x11\n\tclient_id\x18\x04 \x01(\t\x12\x0f\n\x07\x61uth_id\x18\x05 \x01(\t\x12\x11\n\ttoken_url\x18\n \x01(\t\x12\x15\n\rclient_secret\x18\x0b \x01(\t\x1a%\n\x06\x41PIKey\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\x0e\n\x06rotate\x18\x02 \x01(\x08\x1a\x16\n\x03TLS\x12\x0f\n\x07subject\x18\x01 \x01(\tB\x0b\n\t_disabled\"8\n\rCreateRequest\x12\'\n\x06\x63lient\x18\x01 \x01(\x0b\x32\x17.content.clients.Client\":\n\x0e\x43reateResponse\x12(\n\x07\x63reated\x18\x01 \x01(\x0b\x32\x17.content.clients.Client\"*\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\n\n\x02id\x18\x02 \x01(\t\"6\n\x0bGetResponse\x12\'\n\x06\x63lient\x18\x01 \x01(\x0b\x32\x17.content.clients.Client\"_\n\x0cGetByRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x17\n\x0foauth_client_id\x18\x02 \x01(\t\x12\x13\n\x0btls_subject\x18\x03 \x01(\t\x12\x0f\n\x07\x61pi_key\x18\x04 \x01(\t\"8\n\rGetByResponse\x12\'\n\x06\x63lient\x18\x01 \x01(\x0b\x32\x17.content.clients.Client\"8\n\rUpdateRequest\x12\'\n\x06\x63lient\x18\x01 \x01(\x0b\x32\x17.content.clients.Client\"\x1f\n\x0bListRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"8\n\x0cListResponse\x12(\n\x07\x63lients\x18\x01 \x03(\x0b\x32\x17.content.clients.Client\"-\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\n\n\x02id\x18\x02 \x01(\t\"=\n\rEnableRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\n\n\x02id\x18\x02 \x01(\t\x12\x0e\n\x06\x65nable\x18\x03 \x01(\x08\x32\xf7\x03\n\x07\x43lients\x12K\n\x06\x43reate\x12\x1e.content.clients.CreateRequest\x1a\x1f.content.clients.CreateResponse\"\x00\x12\x42\n\x03Get\x12\x1b.content.clients.GetRequest\x1a\x1c.content.clients.GetResponse\"\x00\x12H\n\x05GetBy\x12\x1d.content.clients.GetByRequest\x1a\x1e.content.clients.GetByResponse\"\x00\x12\x42\n\x06Update\x12\x1e.content.clients.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x45\n\x04List\x12\x1c.content.clients.ListRequest\x1a\x1d.content.clients.ListResponse\"\x00\x12\x42\n\x06\x44\x65lete\x12\x1e.content.clients.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x42\n\x06\x45nable\x12\x1e.content.clients.EnableRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x34Z2git.perx.ru/perxis/perxis-go/proto/clients;clientsb\x06proto3')
+
+
+
+_CLIENT = DESCRIPTOR.message_types_by_name['Client']
+_CLIENT_OAUTH = _CLIENT.nested_types_by_name['OAuth']
+_CLIENT_APIKEY = _CLIENT.nested_types_by_name['APIKey']
+_CLIENT_TLS = _CLIENT.nested_types_by_name['TLS']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_GETBYREQUEST = DESCRIPTOR.message_types_by_name['GetByRequest']
+_GETBYRESPONSE = DESCRIPTOR.message_types_by_name['GetByResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_ENABLEREQUEST = DESCRIPTOR.message_types_by_name['EnableRequest']
+Client = _reflection.GeneratedProtocolMessageType('Client', (_message.Message,), {
+
+  'OAuth' : _reflection.GeneratedProtocolMessageType('OAuth', (_message.Message,), {
+    'DESCRIPTOR' : _CLIENT_OAUTH,
+    '__module__' : 'clients.clients_pb2'
+    # @@protoc_insertion_point(class_scope:content.clients.Client.OAuth)
+    })
+  ,
+
+  'APIKey' : _reflection.GeneratedProtocolMessageType('APIKey', (_message.Message,), {
+    'DESCRIPTOR' : _CLIENT_APIKEY,
+    '__module__' : 'clients.clients_pb2'
+    # @@protoc_insertion_point(class_scope:content.clients.Client.APIKey)
+    })
+  ,
+
+  'TLS' : _reflection.GeneratedProtocolMessageType('TLS', (_message.Message,), {
+    'DESCRIPTOR' : _CLIENT_TLS,
+    '__module__' : 'clients.clients_pb2'
+    # @@protoc_insertion_point(class_scope:content.clients.Client.TLS)
+    })
+  ,
+  'DESCRIPTOR' : _CLIENT,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.Client)
+  })
+_sym_db.RegisterMessage(Client)
+_sym_db.RegisterMessage(Client.OAuth)
+_sym_db.RegisterMessage(Client.APIKey)
+_sym_db.RegisterMessage(Client.TLS)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+GetByRequest = _reflection.GeneratedProtocolMessageType('GetByRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETBYREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.GetByRequest)
+  })
+_sym_db.RegisterMessage(GetByRequest)
+
+GetByResponse = _reflection.GeneratedProtocolMessageType('GetByResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETBYRESPONSE,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.GetByResponse)
+  })
+_sym_db.RegisterMessage(GetByResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+EnableRequest = _reflection.GeneratedProtocolMessageType('EnableRequest', (_message.Message,), {
+  'DESCRIPTOR' : _ENABLEREQUEST,
+  '__module__' : 'clients.clients_pb2'
+  # @@protoc_insertion_point(class_scope:content.clients.EnableRequest)
+  })
+_sym_db.RegisterMessage(EnableRequest)
+
+_CLIENTS = DESCRIPTOR.services_by_name['Clients']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z2git.perx.ru/perxis/perxis-go/proto/clients;clients'
+  _CLIENT._serialized_start=72
+  _CLIENT._serialized_end=485
+  _CLIENT_OAUTH._serialized_start=324
+  _CLIENT_OAUTH._serialized_end=409
+  _CLIENT_APIKEY._serialized_start=411
+  _CLIENT_APIKEY._serialized_end=448
+  _CLIENT_TLS._serialized_start=450
+  _CLIENT_TLS._serialized_end=472
+  _CREATEREQUEST._serialized_start=487
+  _CREATEREQUEST._serialized_end=543
+  _CREATERESPONSE._serialized_start=545
+  _CREATERESPONSE._serialized_end=603
+  _GETREQUEST._serialized_start=605
+  _GETREQUEST._serialized_end=647
+  _GETRESPONSE._serialized_start=649
+  _GETRESPONSE._serialized_end=703
+  _GETBYREQUEST._serialized_start=705
+  _GETBYREQUEST._serialized_end=800
+  _GETBYRESPONSE._serialized_start=802
+  _GETBYRESPONSE._serialized_end=858
+  _UPDATEREQUEST._serialized_start=860
+  _UPDATEREQUEST._serialized_end=916
+  _LISTREQUEST._serialized_start=918
+  _LISTREQUEST._serialized_end=949
+  _LISTRESPONSE._serialized_start=951
+  _LISTRESPONSE._serialized_end=1007
+  _DELETEREQUEST._serialized_start=1009
+  _DELETEREQUEST._serialized_end=1054
+  _ENABLEREQUEST._serialized_start=1056
+  _ENABLEREQUEST._serialized_end=1117
+  _CLIENTS._serialized_start=1120
+  _CLIENTS._serialized_end=1623
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/clients/clients_pb2_grpc.py b/clients/clients/clients_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..c699bf243bea7a68ae89b669b46ba9b82001e860
--- /dev/null
+++ b/clients/clients/clients_pb2_grpc.py
@@ -0,0 +1,272 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from clients import clients_pb2 as clients_dot_clients__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+class ClientsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.clients.Clients/Create',
+                request_serializer=clients_dot_clients__pb2.CreateRequest.SerializeToString,
+                response_deserializer=clients_dot_clients__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.clients.Clients/Get',
+                request_serializer=clients_dot_clients__pb2.GetRequest.SerializeToString,
+                response_deserializer=clients_dot_clients__pb2.GetResponse.FromString,
+                )
+        self.GetBy = channel.unary_unary(
+                '/content.clients.Clients/GetBy',
+                request_serializer=clients_dot_clients__pb2.GetByRequest.SerializeToString,
+                response_deserializer=clients_dot_clients__pb2.GetByResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.clients.Clients/Update',
+                request_serializer=clients_dot_clients__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.clients.Clients/List',
+                request_serializer=clients_dot_clients__pb2.ListRequest.SerializeToString,
+                response_deserializer=clients_dot_clients__pb2.ListResponse.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.clients.Clients/Delete',
+                request_serializer=clients_dot_clients__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Enable = channel.unary_unary(
+                '/content.clients.Clients/Enable',
+                request_serializer=clients_dot_clients__pb2.EnableRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class ClientsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Create - создает клиента (приложение) для работы с API
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Get - возвращает клиента по id
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetBy(self, request, context):
+        """GetByClientID - возвращает клиента по clientId системы авторизации
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Update - обновляет параметры клиента
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """List - возвращает список клиентов созданных в пространстве
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Delete - удаляет указанного клиента из пространстве
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Enable(self, request, context):
+        """Enable - активирует/деактивирует клиента. Клиент не сможет обращаться к API платформы
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ClientsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=clients_dot_clients__pb2.CreateRequest.FromString,
+                    response_serializer=clients_dot_clients__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=clients_dot_clients__pb2.GetRequest.FromString,
+                    response_serializer=clients_dot_clients__pb2.GetResponse.SerializeToString,
+            ),
+            'GetBy': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetBy,
+                    request_deserializer=clients_dot_clients__pb2.GetByRequest.FromString,
+                    response_serializer=clients_dot_clients__pb2.GetByResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=clients_dot_clients__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=clients_dot_clients__pb2.ListRequest.FromString,
+                    response_serializer=clients_dot_clients__pb2.ListResponse.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=clients_dot_clients__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Enable': grpc.unary_unary_rpc_method_handler(
+                    servicer.Enable,
+                    request_deserializer=clients_dot_clients__pb2.EnableRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.clients.Clients', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Clients(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/Create',
+            clients_dot_clients__pb2.CreateRequest.SerializeToString,
+            clients_dot_clients__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/Get',
+            clients_dot_clients__pb2.GetRequest.SerializeToString,
+            clients_dot_clients__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetBy(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/GetBy',
+            clients_dot_clients__pb2.GetByRequest.SerializeToString,
+            clients_dot_clients__pb2.GetByResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/Update',
+            clients_dot_clients__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/List',
+            clients_dot_clients__pb2.ListRequest.SerializeToString,
+            clients_dot_clients__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/Delete',
+            clients_dot_clients__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Enable(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.clients.Clients/Enable',
+            clients_dot_clients__pb2.EnableRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/collaborators/collaborators_pb2.py b/clients/collaborators/collaborators_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..e522a44ffe82e4e0c9e6603e2da9e2dc4cc104db
--- /dev/null
+++ b/clients/collaborators/collaborators_pb2.py
@@ -0,0 +1,110 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: collaborators/collaborators.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n!collaborators/collaborators.proto\x12\x15\x63ontent.collaborators\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"=\n\nSetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07subject\x18\x02 \x01(\t\x12\x0c\n\x04role\x18\x03 \x01(\t\"/\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07subject\x18\x02 \x01(\t\"\x1b\n\x0bGetResponse\x12\x0c\n\x04role\x18\x03 \x01(\t\"2\n\rRemoveRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07subject\x18\x02 \x01(\t\",\n\x18ListCollaboratorsRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"H\n\x19ListCollaboratorsResponse\x12+\n\rcollaborators\x18\x01 \x03(\x0b\x32\x14.common.Collaborator\"$\n\x11ListSpacesRequest\x12\x0f\n\x07subject\x18\x01 \x01(\t\":\n\x12ListSpacesResponse\x12$\n\x06spaces\x18\x01 \x03(\x0b\x32\x14.common.Collaborator2\xcc\x03\n\rCollaborators\x12\x42\n\x03Set\x12!.content.collaborators.SetRequest\x1a\x16.google.protobuf.Empty\"\x00\x12N\n\x03Get\x12!.content.collaborators.GetRequest\x1a\".content.collaborators.GetResponse\"\x00\x12H\n\x06Remove\x12$.content.collaborators.RemoveRequest\x1a\x16.google.protobuf.Empty\"\x00\x12x\n\x11ListCollaborators\x12/.content.collaborators.ListCollaboratorsRequest\x1a\x30.content.collaborators.ListCollaboratorsResponse\"\x00\x12\x63\n\nListSpaces\x12(.content.collaborators.ListSpacesRequest\x1a).content.collaborators.ListSpacesResponse\"\x00\x42@Z>git.perx.ru/perxis/perxis-go/proto/collaborators;collaboratorsb\x06proto3')
+
+
+
+_SETREQUEST = DESCRIPTOR.message_types_by_name['SetRequest']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_REMOVEREQUEST = DESCRIPTOR.message_types_by_name['RemoveRequest']
+_LISTCOLLABORATORSREQUEST = DESCRIPTOR.message_types_by_name['ListCollaboratorsRequest']
+_LISTCOLLABORATORSRESPONSE = DESCRIPTOR.message_types_by_name['ListCollaboratorsResponse']
+_LISTSPACESREQUEST = DESCRIPTOR.message_types_by_name['ListSpacesRequest']
+_LISTSPACESRESPONSE = DESCRIPTOR.message_types_by_name['ListSpacesResponse']
+SetRequest = _reflection.GeneratedProtocolMessageType('SetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _SETREQUEST,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.SetRequest)
+  })
+_sym_db.RegisterMessage(SetRequest)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+RemoveRequest = _reflection.GeneratedProtocolMessageType('RemoveRequest', (_message.Message,), {
+  'DESCRIPTOR' : _REMOVEREQUEST,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.RemoveRequest)
+  })
+_sym_db.RegisterMessage(RemoveRequest)
+
+ListCollaboratorsRequest = _reflection.GeneratedProtocolMessageType('ListCollaboratorsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTCOLLABORATORSREQUEST,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.ListCollaboratorsRequest)
+  })
+_sym_db.RegisterMessage(ListCollaboratorsRequest)
+
+ListCollaboratorsResponse = _reflection.GeneratedProtocolMessageType('ListCollaboratorsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTCOLLABORATORSRESPONSE,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.ListCollaboratorsResponse)
+  })
+_sym_db.RegisterMessage(ListCollaboratorsResponse)
+
+ListSpacesRequest = _reflection.GeneratedProtocolMessageType('ListSpacesRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTSPACESREQUEST,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.ListSpacesRequest)
+  })
+_sym_db.RegisterMessage(ListSpacesRequest)
+
+ListSpacesResponse = _reflection.GeneratedProtocolMessageType('ListSpacesResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTSPACESRESPONSE,
+  '__module__' : 'collaborators.collaborators_pb2'
+  # @@protoc_insertion_point(class_scope:content.collaborators.ListSpacesResponse)
+  })
+_sym_db.RegisterMessage(ListSpacesResponse)
+
+_COLLABORATORS = DESCRIPTOR.services_by_name['Collaborators']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z>git.perx.ru/perxis/perxis-go/proto/collaborators;collaborators'
+  _SETREQUEST._serialized_start=110
+  _SETREQUEST._serialized_end=171
+  _GETREQUEST._serialized_start=173
+  _GETREQUEST._serialized_end=220
+  _GETRESPONSE._serialized_start=222
+  _GETRESPONSE._serialized_end=249
+  _REMOVEREQUEST._serialized_start=251
+  _REMOVEREQUEST._serialized_end=301
+  _LISTCOLLABORATORSREQUEST._serialized_start=303
+  _LISTCOLLABORATORSREQUEST._serialized_end=347
+  _LISTCOLLABORATORSRESPONSE._serialized_start=349
+  _LISTCOLLABORATORSRESPONSE._serialized_end=421
+  _LISTSPACESREQUEST._serialized_start=423
+  _LISTSPACESREQUEST._serialized_end=459
+  _LISTSPACESRESPONSE._serialized_start=461
+  _LISTSPACESRESPONSE._serialized_end=519
+  _COLLABORATORS._serialized_start=522
+  _COLLABORATORS._serialized_end=982
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/collaborators/collaborators_pb2_grpc.py b/clients/collaborators/collaborators_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..b7f685c6dc93816dc4e8369880b31f625571ef92
--- /dev/null
+++ b/clients/collaborators/collaborators_pb2_grpc.py
@@ -0,0 +1,199 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from collaborators import collaborators_pb2 as collaborators_dot_collaborators__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+class CollaboratorsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Set = channel.unary_unary(
+                '/content.collaborators.Collaborators/Set',
+                request_serializer=collaborators_dot_collaborators__pb2.SetRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.collaborators.Collaborators/Get',
+                request_serializer=collaborators_dot_collaborators__pb2.GetRequest.SerializeToString,
+                response_deserializer=collaborators_dot_collaborators__pb2.GetResponse.FromString,
+                )
+        self.Remove = channel.unary_unary(
+                '/content.collaborators.Collaborators/Remove',
+                request_serializer=collaborators_dot_collaborators__pb2.RemoveRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.ListCollaborators = channel.unary_unary(
+                '/content.collaborators.Collaborators/ListCollaborators',
+                request_serializer=collaborators_dot_collaborators__pb2.ListCollaboratorsRequest.SerializeToString,
+                response_deserializer=collaborators_dot_collaborators__pb2.ListCollaboratorsResponse.FromString,
+                )
+        self.ListSpaces = channel.unary_unary(
+                '/content.collaborators.Collaborators/ListSpaces',
+                request_serializer=collaborators_dot_collaborators__pb2.ListSpacesRequest.SerializeToString,
+                response_deserializer=collaborators_dot_collaborators__pb2.ListSpacesResponse.FromString,
+                )
+
+
+class CollaboratorsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Set(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Remove(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListCollaborators(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListSpaces(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_CollaboratorsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Set': grpc.unary_unary_rpc_method_handler(
+                    servicer.Set,
+                    request_deserializer=collaborators_dot_collaborators__pb2.SetRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=collaborators_dot_collaborators__pb2.GetRequest.FromString,
+                    response_serializer=collaborators_dot_collaborators__pb2.GetResponse.SerializeToString,
+            ),
+            'Remove': grpc.unary_unary_rpc_method_handler(
+                    servicer.Remove,
+                    request_deserializer=collaborators_dot_collaborators__pb2.RemoveRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'ListCollaborators': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListCollaborators,
+                    request_deserializer=collaborators_dot_collaborators__pb2.ListCollaboratorsRequest.FromString,
+                    response_serializer=collaborators_dot_collaborators__pb2.ListCollaboratorsResponse.SerializeToString,
+            ),
+            'ListSpaces': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListSpaces,
+                    request_deserializer=collaborators_dot_collaborators__pb2.ListSpacesRequest.FromString,
+                    response_serializer=collaborators_dot_collaborators__pb2.ListSpacesResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.collaborators.Collaborators', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Collaborators(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Set(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collaborators.Collaborators/Set',
+            collaborators_dot_collaborators__pb2.SetRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collaborators.Collaborators/Get',
+            collaborators_dot_collaborators__pb2.GetRequest.SerializeToString,
+            collaborators_dot_collaborators__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Remove(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collaborators.Collaborators/Remove',
+            collaborators_dot_collaborators__pb2.RemoveRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListCollaborators(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collaborators.Collaborators/ListCollaborators',
+            collaborators_dot_collaborators__pb2.ListCollaboratorsRequest.SerializeToString,
+            collaborators_dot_collaborators__pb2.ListCollaboratorsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListSpaces(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collaborators.Collaborators/ListSpaces',
+            collaborators_dot_collaborators__pb2.ListSpacesRequest.SerializeToString,
+            collaborators_dot_collaborators__pb2.ListSpacesResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/collections/collections_pb2.py b/clients/collections/collections_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..16c2ac27f7873a9286e4343620f65ab4948c3dc0
--- /dev/null
+++ b/clients/collections/collections_pb2.py
@@ -0,0 +1,187 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: collections/collections.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+from google.protobuf import timestamp_pb2 as google_dot_protobuf_dot_timestamp__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1d\x63ollections/collections.proto\x12\x13\x63ontent.collections\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\x1a\x1fgoogle/protobuf/timestamp.proto\"s\n\x06\x41\x63\x63\x65ss\x12\x1f\n\x07\x61\x63tions\x18\x01 \x03(\x0e\x32\x0e.common.Action\x12\x15\n\rhidden_fields\x18\x05 \x03(\t\x12\x17\n\x0freadonly_fields\x18\x06 \x03(\t\x12\x18\n\x10writeonly_fields\x18\x07 \x03(\t\"\x80\x05\n\nCollection\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x03 \x01(\t\x12\x0c\n\x04name\x18\x04 \x01(\t\x12\x13\n\x06single\x18\x05 \x01(\x08H\x00\x88\x01\x01\x12\x13\n\x06system\x18\x06 \x01(\x08H\x01\x88\x01\x01\x12\x14\n\x07no_data\x18\x0c \x01(\x08H\x02\x88\x01\x01\x12\x0e\n\x06schema\x18\x07 \x01(\t\x12\x0e\n\x06hidden\x18\r \x01(\x08\x12\x32\n\x04view\x18\t \x01(\x0b\x32$.content.collections.Collection.View\x12=\n\nstate_info\x18\n \x01(\x0b\x32).content.collections.Collection.StateInfo\x12+\n\x06\x61\x63\x63\x65ss\x18\x14 \x01(\x0b\x32\x1b.content.collections.Access\x1aO\n\x04View\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0e\n\x06\x66ilter\x18\x04 \x01(\t\x1a\x7f\n\tStateInfo\x12\x34\n\x05state\x18\x01 \x01(\x0e\x32%.content.collections.Collection.State\x12\x0c\n\x04info\x18\x02 \x01(\t\x12.\n\nstarted_at\x18\x03 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\"B\n\x05State\x12\x07\n\x03NEW\x10\x00\x12\r\n\tPREPARING\x10\x01\x12\t\n\x05READY\x10\x02\x12\t\n\x05\x45RROR\x10\x03\x12\x0b\n\x07\x43HANGED\x10\x04\x42\t\n\x07_singleB\t\n\x07_systemB\n\n\x08_no_data\"D\n\rCreateRequest\x12\x33\n\ncollection\x18\x01 \x01(\x0b\x32\x1f.content.collections.Collection\"B\n\x0e\x43reateResponse\x12\x30\n\x07\x63reated\x18\x01 \x01(\x0b\x32\x1f.content.collections.Collection\"-\n\nGetOptions\x12\x1f\n\x17\x64isable_schema_includes\x18\x01 \x01(\x08\"w\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x30\n\x07options\x18\x04 \x01(\x0b\x32\x1f.content.collections.GetOptions\"B\n\x0bGetResponse\x12\x33\n\ncollection\x18\x01 \x01(\x0b\x32\x1f.content.collections.Collection\"\xd5\x01\n\x0bListRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x37\n\x06\x66ilter\x18\x05 \x01(\x0b\x32\'.content.collections.ListRequest.Filter\x1ak\n\x06\x46ilter\x12\x16\n\x0e\x65xclude_system\x18\x01 \x01(\x08\x12\x17\n\x0finclude_no_data\x18\x05 \x01(\x08\x12\x16\n\x0einclude_hidden\x18\x06 \x01(\x08\x12\x0c\n\x04name\x18\x03 \x03(\t\x12\n\n\x02id\x18\x04 \x03(\t\"D\n\x0cListResponse\x12\x34\n\x0b\x63ollections\x18\x01 \x03(\x0b\x32\x1f.content.collections.Collection\"D\n\rUpdateRequest\x12\x33\n\ncollection\x18\x01 \x01(\x0b\x32\x1f.content.collections.Collection\"[\n\x10SetSchemaRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0e\n\x06schema\x18\x04 \x01(\t\"H\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t2\xdb\x03\n\x0b\x43ollections\x12S\n\x06\x43reate\x12\".content.collections.CreateRequest\x1a#.content.collections.CreateResponse\"\x00\x12J\n\x03Get\x12\x1f.content.collections.GetRequest\x1a .content.collections.GetResponse\"\x00\x12M\n\x04List\x12 .content.collections.ListRequest\x1a!.content.collections.ListResponse\"\x00\x12\x46\n\x06Update\x12\".content.collections.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12L\n\tSetSchema\x12%.content.collections.SetSchemaRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x46\n\x06\x44\x65lete\x12\".content.collections.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42<Z:git.perx.ru/perxis/perxis-go/proto/collections;collectionsb\x06proto3')
+
+
+
+_ACCESS = DESCRIPTOR.message_types_by_name['Access']
+_COLLECTION = DESCRIPTOR.message_types_by_name['Collection']
+_COLLECTION_VIEW = _COLLECTION.nested_types_by_name['View']
+_COLLECTION_STATEINFO = _COLLECTION.nested_types_by_name['StateInfo']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETOPTIONS = DESCRIPTOR.message_types_by_name['GetOptions']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTREQUEST_FILTER = _LISTREQUEST.nested_types_by_name['Filter']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_SETSCHEMAREQUEST = DESCRIPTOR.message_types_by_name['SetSchemaRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_COLLECTION_STATE = _COLLECTION.enum_types_by_name['State']
+Access = _reflection.GeneratedProtocolMessageType('Access', (_message.Message,), {
+  'DESCRIPTOR' : _ACCESS,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.Access)
+  })
+_sym_db.RegisterMessage(Access)
+
+Collection = _reflection.GeneratedProtocolMessageType('Collection', (_message.Message,), {
+
+  'View' : _reflection.GeneratedProtocolMessageType('View', (_message.Message,), {
+    'DESCRIPTOR' : _COLLECTION_VIEW,
+    '__module__' : 'collections.collections_pb2'
+    # @@protoc_insertion_point(class_scope:content.collections.Collection.View)
+    })
+  ,
+
+  'StateInfo' : _reflection.GeneratedProtocolMessageType('StateInfo', (_message.Message,), {
+    'DESCRIPTOR' : _COLLECTION_STATEINFO,
+    '__module__' : 'collections.collections_pb2'
+    # @@protoc_insertion_point(class_scope:content.collections.Collection.StateInfo)
+    })
+  ,
+  'DESCRIPTOR' : _COLLECTION,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.Collection)
+  })
+_sym_db.RegisterMessage(Collection)
+_sym_db.RegisterMessage(Collection.View)
+_sym_db.RegisterMessage(Collection.StateInfo)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetOptions = _reflection.GeneratedProtocolMessageType('GetOptions', (_message.Message,), {
+  'DESCRIPTOR' : _GETOPTIONS,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.GetOptions)
+  })
+_sym_db.RegisterMessage(GetOptions)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+
+  'Filter' : _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+    'DESCRIPTOR' : _LISTREQUEST_FILTER,
+    '__module__' : 'collections.collections_pb2'
+    # @@protoc_insertion_point(class_scope:content.collections.ListRequest.Filter)
+    })
+  ,
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+_sym_db.RegisterMessage(ListRequest.Filter)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+SetSchemaRequest = _reflection.GeneratedProtocolMessageType('SetSchemaRequest', (_message.Message,), {
+  'DESCRIPTOR' : _SETSCHEMAREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.SetSchemaRequest)
+  })
+_sym_db.RegisterMessage(SetSchemaRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'collections.collections_pb2'
+  # @@protoc_insertion_point(class_scope:content.collections.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+_COLLECTIONS = DESCRIPTOR.services_by_name['Collections']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z:git.perx.ru/perxis/perxis-go/proto/collections;collections'
+  _ACCESS._serialized_start=137
+  _ACCESS._serialized_end=252
+  _COLLECTION._serialized_start=255
+  _COLLECTION._serialized_end=895
+  _COLLECTION_VIEW._serialized_start=585
+  _COLLECTION_VIEW._serialized_end=664
+  _COLLECTION_STATEINFO._serialized_start=666
+  _COLLECTION_STATEINFO._serialized_end=793
+  _COLLECTION_STATE._serialized_start=795
+  _COLLECTION_STATE._serialized_end=861
+  _CREATEREQUEST._serialized_start=897
+  _CREATEREQUEST._serialized_end=965
+  _CREATERESPONSE._serialized_start=967
+  _CREATERESPONSE._serialized_end=1033
+  _GETOPTIONS._serialized_start=1035
+  _GETOPTIONS._serialized_end=1080
+  _GETREQUEST._serialized_start=1082
+  _GETREQUEST._serialized_end=1201
+  _GETRESPONSE._serialized_start=1203
+  _GETRESPONSE._serialized_end=1269
+  _LISTREQUEST._serialized_start=1272
+  _LISTREQUEST._serialized_end=1485
+  _LISTREQUEST_FILTER._serialized_start=1378
+  _LISTREQUEST_FILTER._serialized_end=1485
+  _LISTRESPONSE._serialized_start=1487
+  _LISTRESPONSE._serialized_end=1555
+  _UPDATEREQUEST._serialized_start=1557
+  _UPDATEREQUEST._serialized_end=1625
+  _SETSCHEMAREQUEST._serialized_start=1627
+  _SETSCHEMAREQUEST._serialized_end=1718
+  _DELETEREQUEST._serialized_start=1720
+  _DELETEREQUEST._serialized_end=1792
+  _COLLECTIONS._serialized_start=1795
+  _COLLECTIONS._serialized_end=2270
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/collections/collections_pb2_grpc.py b/clients/collections/collections_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..ec3f4aff5cf9ef0d45b0c1a33b0c2dbbc0ead1ea
--- /dev/null
+++ b/clients/collections/collections_pb2_grpc.py
@@ -0,0 +1,232 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from collections import collections_pb2 as collections_dot_collections__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+class CollectionsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.collections.Collections/Create',
+                request_serializer=collections_dot_collections__pb2.CreateRequest.SerializeToString,
+                response_deserializer=collections_dot_collections__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.collections.Collections/Get',
+                request_serializer=collections_dot_collections__pb2.GetRequest.SerializeToString,
+                response_deserializer=collections_dot_collections__pb2.GetResponse.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.collections.Collections/List',
+                request_serializer=collections_dot_collections__pb2.ListRequest.SerializeToString,
+                response_deserializer=collections_dot_collections__pb2.ListResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.collections.Collections/Update',
+                request_serializer=collections_dot_collections__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.SetSchema = channel.unary_unary(
+                '/content.collections.Collections/SetSchema',
+                request_serializer=collections_dot_collections__pb2.SetSchemaRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.collections.Collections/Delete',
+                request_serializer=collections_dot_collections__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class CollectionsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def SetSchema(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_CollectionsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=collections_dot_collections__pb2.CreateRequest.FromString,
+                    response_serializer=collections_dot_collections__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=collections_dot_collections__pb2.GetRequest.FromString,
+                    response_serializer=collections_dot_collections__pb2.GetResponse.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=collections_dot_collections__pb2.ListRequest.FromString,
+                    response_serializer=collections_dot_collections__pb2.ListResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=collections_dot_collections__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'SetSchema': grpc.unary_unary_rpc_method_handler(
+                    servicer.SetSchema,
+                    request_deserializer=collections_dot_collections__pb2.SetSchemaRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=collections_dot_collections__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.collections.Collections', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Collections(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/Create',
+            collections_dot_collections__pb2.CreateRequest.SerializeToString,
+            collections_dot_collections__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/Get',
+            collections_dot_collections__pb2.GetRequest.SerializeToString,
+            collections_dot_collections__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/List',
+            collections_dot_collections__pb2.ListRequest.SerializeToString,
+            collections_dot_collections__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/Update',
+            collections_dot_collections__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def SetSchema(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/SetSchema',
+            collections_dot_collections__pb2.SetSchemaRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.collections.Collections/Delete',
+            collections_dot_collections__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/common/common_pb2.py b/clients/common/common_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..0ef74a906795044939feb8c875992496c87ea31d
--- /dev/null
+++ b/clients/common/common_pb2.py
@@ -0,0 +1,182 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: common/common.proto
+"""Generated protocol buffer code."""
+from google.protobuf.internal import enum_type_wrapper
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import struct_pb2 as google_dot_protobuf_dot_struct__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x13\x63ommon/common.proto\x12\x06\x63ommon\x1a\x1cgoogle/protobuf/struct.proto\"\xa6\x05\n\x05\x45rror\x12\x12\n\nerror_code\x18\x01 \x01(\x04\x12\x10\n\x08\x65rror_id\x18\x02 \x01(\t\x12\x0e\n\x06reason\x18\x03 \x01(\t\x12\x0e\n\x06\x64omain\x18\x04 \x01(\t\x12-\n\x08metadata\x18\t \x03(\x0b\x32\x1b.common.Error.MetadataEntry\x12-\n\x0b\x62\x61\x64_request\x18\n \x01(\x0b\x32\x18.common.Error.BadRequest\x12+\n\ndebug_info\x18\x0b \x01(\x0b\x32\x17.common.Error.DebugInfo\x12 \n\x04help\x18\x64 \x01(\x0b\x32\x12.common.Error.Help\x12;\n\x12localized_messages\x18\xc8\x01 \x03(\x0b\x32\x1e.common.Error.LocalizedMessage\x1a{\n\nBadRequest\x12\x37\n\x06\x65rrors\x18\x01 \x03(\x0b\x32\'.common.Error.BadRequest.FieldViolation\x1a\x34\n\x0e\x46ieldViolation\x12\r\n\x05\x66ield\x18\x01 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x02 \x01(\t\x1aX\n\x04Help\x12&\n\x05links\x18\x01 \x03(\x0b\x32\x17.common.Error.Help.Link\x1a(\n\x04Link\x12\x13\n\x0b\x64\x65scription\x18\x01 \x01(\t\x12\x0b\n\x03url\x18\x02 \x01(\t\x1a\x30\n\tDebugInfo\x12\x13\n\x0bstack_trace\x18\x01 \x03(\t\x12\x0e\n\x06\x64\x65tail\x18\x02 \x01(\t\x1a\x33\n\x10LocalizedMessage\x12\x0e\n\x06locale\x18\x01 \x01(\t\x12\x0f\n\x07message\x18\x02 \x01(\t\x1a/\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\"J\n\x06\x46ilter\x12\n\n\x02op\x18\x01 \x01(\t\x12\r\n\x05\x66ield\x18\x02 \x01(\t\x12%\n\x05value\x18\x03 \x01(\x0b\x32\x16.google.protobuf.Value\"h\n\x0b\x46indOptions\x12\x0c\n\x04sort\x18\x01 \x03(\t\x12\x10\n\x08page_num\x18\x02 \x01(\x05\x12\x11\n\tpage_size\x18\x03 \x01(\x05\x12\x0e\n\x06\x66ields\x18\x04 \x03(\t\x12\x16\n\x0e\x65xclude_fields\x18\x05 \x01(\x08\"\xd3\x01\n\x04Rule\x12\x15\n\rcollection_id\x18\x01 \x01(\t\x12\x1f\n\x07\x61\x63tions\x18\x02 \x03(\x0e\x32\x0e.common.Action\x12\x1e\n\x06\x61\x63\x63\x65ss\x18\x03 \x01(\x0e\x32\x0e.common.Access\x12\x15\n\rhidden_fields\x18\x05 \x03(\t\x12\x17\n\x0freadonly_fields\x18\x06 \x03(\t\x12\x18\n\x10writeonly_fields\x18\x07 \x03(\t\x12\x13\n\x0bread_filter\x18\x08 \x01(\t\x12\x14\n\x0cwrite_filter\x18\t \x01(\t\"?\n\x0c\x43ollaborator\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07subject\x18\x02 \x01(\t\x12\x0c\n\x04role\x18\x03 \x01(\t\"p\n\x07Version\x12\x16\n\x0eserver_version\x18\x01 \x01(\t\x12\x13\n\x0b\x61pi_version\x18\x02 \x01(\t\x12\x12\n\nbuild_time\x18\x03 \x01(\t\x12\x0e\n\x06\x63ommit\x18\x04 \x01(\t\x12\x14\n\x0c\x62uild_number\x18\x05 \x01(\x05*%\n\x06\x41\x63\x63\x65ss\x12\x07\n\x03\x41NY\x10\x00\x12\x08\n\x04MINE\x10\x01\x12\x08\n\x04ROLE\x10\x02*C\n\x06\x41\x63tion\x12\x0b\n\x07UNKNOWN\x10\x00\x12\n\n\x06\x43REATE\x10\x01\x12\x08\n\x04READ\x10\x02\x12\n\n\x06UPDATE\x10\x03\x12\n\n\x06\x44\x45LETE\x10\x04\x42\x32Z0git.perx.ru/perxis/perxis-go/proto/common;commonb\x06proto3')
+
+_ACCESS = DESCRIPTOR.enum_types_by_name['Access']
+Access = enum_type_wrapper.EnumTypeWrapper(_ACCESS)
+_ACTION = DESCRIPTOR.enum_types_by_name['Action']
+Action = enum_type_wrapper.EnumTypeWrapper(_ACTION)
+ANY = 0
+MINE = 1
+ROLE = 2
+UNKNOWN = 0
+CREATE = 1
+READ = 2
+UPDATE = 3
+DELETE = 4
+
+
+_ERROR = DESCRIPTOR.message_types_by_name['Error']
+_ERROR_BADREQUEST = _ERROR.nested_types_by_name['BadRequest']
+_ERROR_BADREQUEST_FIELDVIOLATION = _ERROR_BADREQUEST.nested_types_by_name['FieldViolation']
+_ERROR_HELP = _ERROR.nested_types_by_name['Help']
+_ERROR_HELP_LINK = _ERROR_HELP.nested_types_by_name['Link']
+_ERROR_DEBUGINFO = _ERROR.nested_types_by_name['DebugInfo']
+_ERROR_LOCALIZEDMESSAGE = _ERROR.nested_types_by_name['LocalizedMessage']
+_ERROR_METADATAENTRY = _ERROR.nested_types_by_name['MetadataEntry']
+_FILTER = DESCRIPTOR.message_types_by_name['Filter']
+_FINDOPTIONS = DESCRIPTOR.message_types_by_name['FindOptions']
+_RULE = DESCRIPTOR.message_types_by_name['Rule']
+_COLLABORATOR = DESCRIPTOR.message_types_by_name['Collaborator']
+_VERSION = DESCRIPTOR.message_types_by_name['Version']
+Error = _reflection.GeneratedProtocolMessageType('Error', (_message.Message,), {
+
+  'BadRequest' : _reflection.GeneratedProtocolMessageType('BadRequest', (_message.Message,), {
+
+    'FieldViolation' : _reflection.GeneratedProtocolMessageType('FieldViolation', (_message.Message,), {
+      'DESCRIPTOR' : _ERROR_BADREQUEST_FIELDVIOLATION,
+      '__module__' : 'common.common_pb2'
+      # @@protoc_insertion_point(class_scope:common.Error.BadRequest.FieldViolation)
+      })
+    ,
+    'DESCRIPTOR' : _ERROR_BADREQUEST,
+    '__module__' : 'common.common_pb2'
+    # @@protoc_insertion_point(class_scope:common.Error.BadRequest)
+    })
+  ,
+
+  'Help' : _reflection.GeneratedProtocolMessageType('Help', (_message.Message,), {
+
+    'Link' : _reflection.GeneratedProtocolMessageType('Link', (_message.Message,), {
+      'DESCRIPTOR' : _ERROR_HELP_LINK,
+      '__module__' : 'common.common_pb2'
+      # @@protoc_insertion_point(class_scope:common.Error.Help.Link)
+      })
+    ,
+    'DESCRIPTOR' : _ERROR_HELP,
+    '__module__' : 'common.common_pb2'
+    # @@protoc_insertion_point(class_scope:common.Error.Help)
+    })
+  ,
+
+  'DebugInfo' : _reflection.GeneratedProtocolMessageType('DebugInfo', (_message.Message,), {
+    'DESCRIPTOR' : _ERROR_DEBUGINFO,
+    '__module__' : 'common.common_pb2'
+    # @@protoc_insertion_point(class_scope:common.Error.DebugInfo)
+    })
+  ,
+
+  'LocalizedMessage' : _reflection.GeneratedProtocolMessageType('LocalizedMessage', (_message.Message,), {
+    'DESCRIPTOR' : _ERROR_LOCALIZEDMESSAGE,
+    '__module__' : 'common.common_pb2'
+    # @@protoc_insertion_point(class_scope:common.Error.LocalizedMessage)
+    })
+  ,
+
+  'MetadataEntry' : _reflection.GeneratedProtocolMessageType('MetadataEntry', (_message.Message,), {
+    'DESCRIPTOR' : _ERROR_METADATAENTRY,
+    '__module__' : 'common.common_pb2'
+    # @@protoc_insertion_point(class_scope:common.Error.MetadataEntry)
+    })
+  ,
+  'DESCRIPTOR' : _ERROR,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.Error)
+  })
+_sym_db.RegisterMessage(Error)
+_sym_db.RegisterMessage(Error.BadRequest)
+_sym_db.RegisterMessage(Error.BadRequest.FieldViolation)
+_sym_db.RegisterMessage(Error.Help)
+_sym_db.RegisterMessage(Error.Help.Link)
+_sym_db.RegisterMessage(Error.DebugInfo)
+_sym_db.RegisterMessage(Error.LocalizedMessage)
+_sym_db.RegisterMessage(Error.MetadataEntry)
+
+Filter = _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+  'DESCRIPTOR' : _FILTER,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.Filter)
+  })
+_sym_db.RegisterMessage(Filter)
+
+FindOptions = _reflection.GeneratedProtocolMessageType('FindOptions', (_message.Message,), {
+  'DESCRIPTOR' : _FINDOPTIONS,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.FindOptions)
+  })
+_sym_db.RegisterMessage(FindOptions)
+
+Rule = _reflection.GeneratedProtocolMessageType('Rule', (_message.Message,), {
+  'DESCRIPTOR' : _RULE,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.Rule)
+  })
+_sym_db.RegisterMessage(Rule)
+
+Collaborator = _reflection.GeneratedProtocolMessageType('Collaborator', (_message.Message,), {
+  'DESCRIPTOR' : _COLLABORATOR,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.Collaborator)
+  })
+_sym_db.RegisterMessage(Collaborator)
+
+Version = _reflection.GeneratedProtocolMessageType('Version', (_message.Message,), {
+  'DESCRIPTOR' : _VERSION,
+  '__module__' : 'common.common_pb2'
+  # @@protoc_insertion_point(class_scope:common.Version)
+  })
+_sym_db.RegisterMessage(Version)
+
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z0git.perx.ru/perxis/perxis-go/proto/common;common'
+  _ERROR_METADATAENTRY._options = None
+  _ERROR_METADATAENTRY._serialized_options = b'8\001'
+  _ACCESS._serialized_start=1317
+  _ACCESS._serialized_end=1354
+  _ACTION._serialized_start=1356
+  _ACTION._serialized_end=1423
+  _ERROR._serialized_start=62
+  _ERROR._serialized_end=740
+  _ERROR_BADREQUEST._serialized_start=375
+  _ERROR_BADREQUEST._serialized_end=498
+  _ERROR_BADREQUEST_FIELDVIOLATION._serialized_start=446
+  _ERROR_BADREQUEST_FIELDVIOLATION._serialized_end=498
+  _ERROR_HELP._serialized_start=500
+  _ERROR_HELP._serialized_end=588
+  _ERROR_HELP_LINK._serialized_start=548
+  _ERROR_HELP_LINK._serialized_end=588
+  _ERROR_DEBUGINFO._serialized_start=590
+  _ERROR_DEBUGINFO._serialized_end=638
+  _ERROR_LOCALIZEDMESSAGE._serialized_start=640
+  _ERROR_LOCALIZEDMESSAGE._serialized_end=691
+  _ERROR_METADATAENTRY._serialized_start=693
+  _ERROR_METADATAENTRY._serialized_end=740
+  _FILTER._serialized_start=742
+  _FILTER._serialized_end=816
+  _FINDOPTIONS._serialized_start=818
+  _FINDOPTIONS._serialized_end=922
+  _RULE._serialized_start=925
+  _RULE._serialized_end=1136
+  _COLLABORATOR._serialized_start=1138
+  _COLLABORATOR._serialized_end=1201
+  _VERSION._serialized_start=1203
+  _VERSION._serialized_end=1315
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/common/common_pb2_grpc.py b/clients/common/common_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..2daafffebfc817aefe8fcb96eaec25e65b3903e8
--- /dev/null
+++ b/clients/common/common_pb2_grpc.py
@@ -0,0 +1,4 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
diff --git a/clients/delivery/delivery_pb2.py b/clients/delivery/delivery_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..3ffb52d1cfd2f2a9eca28bf66f2255bcc87226c9
--- /dev/null
+++ b/clients/delivery/delivery_pb2.py
@@ -0,0 +1,172 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: delivery/delivery.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from locales import locales_pb2 as locales_dot_locales__pb2
+from environments import environments_pb2 as environments_dot_environments__pb2
+from collections import collections_pb2 as collections_dot_collections__pb2
+from items import items_pb2 as items_dot_items__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x17\x64\x65livery/delivery.proto\x12\x08\x64\x65livery\x1a\x15locales/locales.proto\x1a\x1f\x65nvironments/environments.proto\x1a\x1d\x63ollections/collections.proto\x1a\x11items/items.proto\"&\n\x12ListLocalesRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"?\n\x13ListLocalesResponse\x12(\n\x07locales\x18\x01 \x03(\x0b\x32\x17.content.locales.Locale\"9\n\x15GetEnvironmentRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\"H\n\x16GetEnvironmentResponse\x12.\n\x03\x65nv\x18\x01 \x01(\x0b\x32!.content.environments.Environment\"+\n\x17ListEnvironmentsRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"K\n\x18ListEnvironmentsResponse\x12/\n\x04\x65nvs\x18\x01 \x03(\x0b\x32!.content.environments.Environment\"O\n\x14GetCollectionRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\"L\n\x15GetCollectionResponse\x12\x33\n\ncollection\x18\x01 \x01(\x0b\x32\x1f.content.collections.Collection\":\n\x16ListCollectionsRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\"O\n\x17ListCollectionsResponse\x12\x34\n\x0b\x63ollections\x18\x01 \x03(\x0b\x32\x1f.content.collections.Collection\"\x8f\x01\n\x0eGetItemRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x05 \x01(\t\x12\x33\n\x07options\x18\x06 \x01(\x0b\x32\".content.items.GetPublishedOptions\"4\n\x0fGetItemResponse\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\xa8\x01\n\x10\x46indItemsRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12%\n\x06\x66ilter\x18\x05 \x01(\x0b\x32\x15.content.items.Filter\x12\x34\n\x07options\x18\x06 \x01(\x0b\x32#.content.items.FindPublishedOptions\"F\n\x11\x46indItemsResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\x12\r\n\x05total\x18\x02 \x01(\x05\x32\xc4\x04\n\x08\x44\x65livery\x12L\n\x0bListLocales\x12\x1c.delivery.ListLocalesRequest\x1a\x1d.delivery.ListLocalesResponse\"\x00\x12U\n\x0eGetEnvironment\x12\x1f.delivery.GetEnvironmentRequest\x1a .delivery.GetEnvironmentResponse\"\x00\x12[\n\x10ListEnvironments\x12!.delivery.ListEnvironmentsRequest\x1a\".delivery.ListEnvironmentsResponse\"\x00\x12R\n\rGetCollection\x12\x1e.delivery.GetCollectionRequest\x1a\x1f.delivery.GetCollectionResponse\"\x00\x12X\n\x0fListCollections\x12 .delivery.ListCollectionsRequest\x1a!.delivery.ListCollectionsResponse\"\x00\x12@\n\x07GetItem\x12\x18.delivery.GetItemRequest\x1a\x19.delivery.GetItemResponse\"\x00\x12\x46\n\tFindItems\x12\x1a.delivery.FindItemsRequest\x1a\x1b.delivery.FindItemsResponse\"\x00\x42\x36Z4git.perx.ru/perxis/perxis-go/proto/delivery;deliveryb\x06proto3')
+
+
+
+_LISTLOCALESREQUEST = DESCRIPTOR.message_types_by_name['ListLocalesRequest']
+_LISTLOCALESRESPONSE = DESCRIPTOR.message_types_by_name['ListLocalesResponse']
+_GETENVIRONMENTREQUEST = DESCRIPTOR.message_types_by_name['GetEnvironmentRequest']
+_GETENVIRONMENTRESPONSE = DESCRIPTOR.message_types_by_name['GetEnvironmentResponse']
+_LISTENVIRONMENTSREQUEST = DESCRIPTOR.message_types_by_name['ListEnvironmentsRequest']
+_LISTENVIRONMENTSRESPONSE = DESCRIPTOR.message_types_by_name['ListEnvironmentsResponse']
+_GETCOLLECTIONREQUEST = DESCRIPTOR.message_types_by_name['GetCollectionRequest']
+_GETCOLLECTIONRESPONSE = DESCRIPTOR.message_types_by_name['GetCollectionResponse']
+_LISTCOLLECTIONSREQUEST = DESCRIPTOR.message_types_by_name['ListCollectionsRequest']
+_LISTCOLLECTIONSRESPONSE = DESCRIPTOR.message_types_by_name['ListCollectionsResponse']
+_GETITEMREQUEST = DESCRIPTOR.message_types_by_name['GetItemRequest']
+_GETITEMRESPONSE = DESCRIPTOR.message_types_by_name['GetItemResponse']
+_FINDITEMSREQUEST = DESCRIPTOR.message_types_by_name['FindItemsRequest']
+_FINDITEMSRESPONSE = DESCRIPTOR.message_types_by_name['FindItemsResponse']
+ListLocalesRequest = _reflection.GeneratedProtocolMessageType('ListLocalesRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTLOCALESREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListLocalesRequest)
+  })
+_sym_db.RegisterMessage(ListLocalesRequest)
+
+ListLocalesResponse = _reflection.GeneratedProtocolMessageType('ListLocalesResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTLOCALESRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListLocalesResponse)
+  })
+_sym_db.RegisterMessage(ListLocalesResponse)
+
+GetEnvironmentRequest = _reflection.GeneratedProtocolMessageType('GetEnvironmentRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETENVIRONMENTREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetEnvironmentRequest)
+  })
+_sym_db.RegisterMessage(GetEnvironmentRequest)
+
+GetEnvironmentResponse = _reflection.GeneratedProtocolMessageType('GetEnvironmentResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETENVIRONMENTRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetEnvironmentResponse)
+  })
+_sym_db.RegisterMessage(GetEnvironmentResponse)
+
+ListEnvironmentsRequest = _reflection.GeneratedProtocolMessageType('ListEnvironmentsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTENVIRONMENTSREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListEnvironmentsRequest)
+  })
+_sym_db.RegisterMessage(ListEnvironmentsRequest)
+
+ListEnvironmentsResponse = _reflection.GeneratedProtocolMessageType('ListEnvironmentsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTENVIRONMENTSRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListEnvironmentsResponse)
+  })
+_sym_db.RegisterMessage(ListEnvironmentsResponse)
+
+GetCollectionRequest = _reflection.GeneratedProtocolMessageType('GetCollectionRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETCOLLECTIONREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetCollectionRequest)
+  })
+_sym_db.RegisterMessage(GetCollectionRequest)
+
+GetCollectionResponse = _reflection.GeneratedProtocolMessageType('GetCollectionResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETCOLLECTIONRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetCollectionResponse)
+  })
+_sym_db.RegisterMessage(GetCollectionResponse)
+
+ListCollectionsRequest = _reflection.GeneratedProtocolMessageType('ListCollectionsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTCOLLECTIONSREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListCollectionsRequest)
+  })
+_sym_db.RegisterMessage(ListCollectionsRequest)
+
+ListCollectionsResponse = _reflection.GeneratedProtocolMessageType('ListCollectionsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTCOLLECTIONSRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.ListCollectionsResponse)
+  })
+_sym_db.RegisterMessage(ListCollectionsResponse)
+
+GetItemRequest = _reflection.GeneratedProtocolMessageType('GetItemRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETITEMREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetItemRequest)
+  })
+_sym_db.RegisterMessage(GetItemRequest)
+
+GetItemResponse = _reflection.GeneratedProtocolMessageType('GetItemResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETITEMRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.GetItemResponse)
+  })
+_sym_db.RegisterMessage(GetItemResponse)
+
+FindItemsRequest = _reflection.GeneratedProtocolMessageType('FindItemsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDITEMSREQUEST,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.FindItemsRequest)
+  })
+_sym_db.RegisterMessage(FindItemsRequest)
+
+FindItemsResponse = _reflection.GeneratedProtocolMessageType('FindItemsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDITEMSRESPONSE,
+  '__module__' : 'delivery.delivery_pb2'
+  # @@protoc_insertion_point(class_scope:delivery.FindItemsResponse)
+  })
+_sym_db.RegisterMessage(FindItemsResponse)
+
+_DELIVERY = DESCRIPTOR.services_by_name['Delivery']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z4git.perx.ru/perxis/perxis-go/proto/delivery;delivery'
+  _LISTLOCALESREQUEST._serialized_start=143
+  _LISTLOCALESREQUEST._serialized_end=181
+  _LISTLOCALESRESPONSE._serialized_start=183
+  _LISTLOCALESRESPONSE._serialized_end=246
+  _GETENVIRONMENTREQUEST._serialized_start=248
+  _GETENVIRONMENTREQUEST._serialized_end=305
+  _GETENVIRONMENTRESPONSE._serialized_start=307
+  _GETENVIRONMENTRESPONSE._serialized_end=379
+  _LISTENVIRONMENTSREQUEST._serialized_start=381
+  _LISTENVIRONMENTSREQUEST._serialized_end=424
+  _LISTENVIRONMENTSRESPONSE._serialized_start=426
+  _LISTENVIRONMENTSRESPONSE._serialized_end=501
+  _GETCOLLECTIONREQUEST._serialized_start=503
+  _GETCOLLECTIONREQUEST._serialized_end=582
+  _GETCOLLECTIONRESPONSE._serialized_start=584
+  _GETCOLLECTIONRESPONSE._serialized_end=660
+  _LISTCOLLECTIONSREQUEST._serialized_start=662
+  _LISTCOLLECTIONSREQUEST._serialized_end=720
+  _LISTCOLLECTIONSRESPONSE._serialized_start=722
+  _LISTCOLLECTIONSRESPONSE._serialized_end=801
+  _GETITEMREQUEST._serialized_start=804
+  _GETITEMREQUEST._serialized_end=947
+  _GETITEMRESPONSE._serialized_start=949
+  _GETITEMRESPONSE._serialized_end=1001
+  _FINDITEMSREQUEST._serialized_start=1004
+  _FINDITEMSREQUEST._serialized_end=1172
+  _FINDITEMSRESPONSE._serialized_start=1174
+  _FINDITEMSRESPONSE._serialized_end=1244
+  _DELIVERY._serialized_start=1247
+  _DELIVERY._serialized_end=1827
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/delivery/delivery_pb2_grpc.py b/clients/delivery/delivery_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..a95af964835926dd307a71c09192b253964bb880
--- /dev/null
+++ b/clients/delivery/delivery_pb2_grpc.py
@@ -0,0 +1,264 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from delivery import delivery_pb2 as delivery_dot_delivery__pb2
+
+
+class DeliveryStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.ListLocales = channel.unary_unary(
+                '/delivery.Delivery/ListLocales',
+                request_serializer=delivery_dot_delivery__pb2.ListLocalesRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.ListLocalesResponse.FromString,
+                )
+        self.GetEnvironment = channel.unary_unary(
+                '/delivery.Delivery/GetEnvironment',
+                request_serializer=delivery_dot_delivery__pb2.GetEnvironmentRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.GetEnvironmentResponse.FromString,
+                )
+        self.ListEnvironments = channel.unary_unary(
+                '/delivery.Delivery/ListEnvironments',
+                request_serializer=delivery_dot_delivery__pb2.ListEnvironmentsRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.ListEnvironmentsResponse.FromString,
+                )
+        self.GetCollection = channel.unary_unary(
+                '/delivery.Delivery/GetCollection',
+                request_serializer=delivery_dot_delivery__pb2.GetCollectionRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.GetCollectionResponse.FromString,
+                )
+        self.ListCollections = channel.unary_unary(
+                '/delivery.Delivery/ListCollections',
+                request_serializer=delivery_dot_delivery__pb2.ListCollectionsRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.ListCollectionsResponse.FromString,
+                )
+        self.GetItem = channel.unary_unary(
+                '/delivery.Delivery/GetItem',
+                request_serializer=delivery_dot_delivery__pb2.GetItemRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.GetItemResponse.FromString,
+                )
+        self.FindItems = channel.unary_unary(
+                '/delivery.Delivery/FindItems',
+                request_serializer=delivery_dot_delivery__pb2.FindItemsRequest.SerializeToString,
+                response_deserializer=delivery_dot_delivery__pb2.FindItemsResponse.FromString,
+                )
+
+
+class DeliveryServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def ListLocales(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetEnvironment(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListEnvironments(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetCollection(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListCollections(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetItem(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def FindItems(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_DeliveryServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'ListLocales': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListLocales,
+                    request_deserializer=delivery_dot_delivery__pb2.ListLocalesRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.ListLocalesResponse.SerializeToString,
+            ),
+            'GetEnvironment': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetEnvironment,
+                    request_deserializer=delivery_dot_delivery__pb2.GetEnvironmentRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.GetEnvironmentResponse.SerializeToString,
+            ),
+            'ListEnvironments': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListEnvironments,
+                    request_deserializer=delivery_dot_delivery__pb2.ListEnvironmentsRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.ListEnvironmentsResponse.SerializeToString,
+            ),
+            'GetCollection': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetCollection,
+                    request_deserializer=delivery_dot_delivery__pb2.GetCollectionRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.GetCollectionResponse.SerializeToString,
+            ),
+            'ListCollections': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListCollections,
+                    request_deserializer=delivery_dot_delivery__pb2.ListCollectionsRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.ListCollectionsResponse.SerializeToString,
+            ),
+            'GetItem': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetItem,
+                    request_deserializer=delivery_dot_delivery__pb2.GetItemRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.GetItemResponse.SerializeToString,
+            ),
+            'FindItems': grpc.unary_unary_rpc_method_handler(
+                    servicer.FindItems,
+                    request_deserializer=delivery_dot_delivery__pb2.FindItemsRequest.FromString,
+                    response_serializer=delivery_dot_delivery__pb2.FindItemsResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'delivery.Delivery', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Delivery(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def ListLocales(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/ListLocales',
+            delivery_dot_delivery__pb2.ListLocalesRequest.SerializeToString,
+            delivery_dot_delivery__pb2.ListLocalesResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetEnvironment(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/GetEnvironment',
+            delivery_dot_delivery__pb2.GetEnvironmentRequest.SerializeToString,
+            delivery_dot_delivery__pb2.GetEnvironmentResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListEnvironments(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/ListEnvironments',
+            delivery_dot_delivery__pb2.ListEnvironmentsRequest.SerializeToString,
+            delivery_dot_delivery__pb2.ListEnvironmentsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetCollection(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/GetCollection',
+            delivery_dot_delivery__pb2.GetCollectionRequest.SerializeToString,
+            delivery_dot_delivery__pb2.GetCollectionResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListCollections(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/ListCollections',
+            delivery_dot_delivery__pb2.ListCollectionsRequest.SerializeToString,
+            delivery_dot_delivery__pb2.ListCollectionsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetItem(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/GetItem',
+            delivery_dot_delivery__pb2.GetItemRequest.SerializeToString,
+            delivery_dot_delivery__pb2.GetItemResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def FindItems(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/delivery.Delivery/FindItems',
+            delivery_dot_delivery__pb2.FindItemsRequest.SerializeToString,
+            delivery_dot_delivery__pb2.FindItemsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/environments/environments_pb2.py b/clients/environments/environments_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..1a9aa1426d092b937e25f6c8fc615df55264b0b5
--- /dev/null
+++ b/clients/environments/environments_pb2.py
@@ -0,0 +1,152 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: environments/environments.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1f\x65nvironments/environments.proto\x12\x14\x63ontent.environments\x1a\x1bgoogle/protobuf/empty.proto\"\x8f\x02\n\x0b\x45nvironment\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x03 \x01(\t\x12\x36\n\x05state\x18\x04 \x01(\x0e\x32\'.content.environments.Environment.State\x12\x12\n\nstate_info\x18\x05 \x01(\t\x12\x0f\n\x07\x61liases\x18\x06 \x03(\t\x12,\n\x06\x63onfig\x18\n \x01(\x0b\x32\x1c.content.environments.Config\"B\n\x05State\x12\x0b\n\x07UNKNOWN\x10\x00\x12\x07\n\x03NEW\x10\x01\x12\r\n\tPREPARING\x10\x02\x12\t\n\x05READY\x10\x03\x12\t\n\x05\x45RROR\x10\x04\"-\n\x06\x43onfig\x12\x11\n\tsource_id\x18\x01 \x01(\t\x12\x10\n\x08\x66\x65\x61tures\x18\x02 \x03(\t\"?\n\rCreateRequest\x12.\n\x03\x65nv\x18\x01 \x01(\x0b\x32!.content.environments.Environment\"D\n\x0e\x43reateResponse\x12\x32\n\x07\x63reated\x18\x01 \x01(\x0b\x32!.content.environments.Environment\"\x1f\n\x0bListRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"?\n\x0cListResponse\x12/\n\x04\x65nvs\x18\x01 \x03(\x0b\x32!.content.environments.Environment\".\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\"=\n\x0bGetResponse\x12.\n\x03\x65nv\x18\x01 \x01(\x0b\x32!.content.environments.Environment\"?\n\rUpdateRequest\x12.\n\x03\x65nv\x18\x01 \x01(\x0b\x32!.content.environments.Environment\"B\n\x0fSetAliasRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\r\n\x05\x61lias\x18\x03 \x01(\t\"E\n\x12RemoveAliasRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\r\n\x05\x61lias\x18\x03 \x01(\t\"1\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t2\xb6\x04\n\x0c\x45nvironments\x12U\n\x06\x43reate\x12#.content.environments.CreateRequest\x1a$.content.environments.CreateResponse\"\x00\x12L\n\x03Get\x12 .content.environments.GetRequest\x1a!.content.environments.GetResponse\"\x00\x12O\n\x04List\x12!.content.environments.ListRequest\x1a\".content.environments.ListResponse\"\x00\x12G\n\x06Update\x12#.content.environments.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12K\n\x08SetAlias\x12%.content.environments.SetAliasRequest\x1a\x16.google.protobuf.Empty\"\x00\x12Q\n\x0bRemoveAlias\x12(.content.environments.RemoveAliasRequest\x1a\x16.google.protobuf.Empty\"\x00\x12G\n\x06\x44\x65lete\x12#.content.environments.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42>Z<git.perx.ru/perxis/perxis-go/proto/environments;environmentsb\x06proto3')
+
+
+
+_ENVIRONMENT = DESCRIPTOR.message_types_by_name['Environment']
+_CONFIG = DESCRIPTOR.message_types_by_name['Config']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_SETALIASREQUEST = DESCRIPTOR.message_types_by_name['SetAliasRequest']
+_REMOVEALIASREQUEST = DESCRIPTOR.message_types_by_name['RemoveAliasRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_ENVIRONMENT_STATE = _ENVIRONMENT.enum_types_by_name['State']
+Environment = _reflection.GeneratedProtocolMessageType('Environment', (_message.Message,), {
+  'DESCRIPTOR' : _ENVIRONMENT,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.Environment)
+  })
+_sym_db.RegisterMessage(Environment)
+
+Config = _reflection.GeneratedProtocolMessageType('Config', (_message.Message,), {
+  'DESCRIPTOR' : _CONFIG,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.Config)
+  })
+_sym_db.RegisterMessage(Config)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+SetAliasRequest = _reflection.GeneratedProtocolMessageType('SetAliasRequest', (_message.Message,), {
+  'DESCRIPTOR' : _SETALIASREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.SetAliasRequest)
+  })
+_sym_db.RegisterMessage(SetAliasRequest)
+
+RemoveAliasRequest = _reflection.GeneratedProtocolMessageType('RemoveAliasRequest', (_message.Message,), {
+  'DESCRIPTOR' : _REMOVEALIASREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.RemoveAliasRequest)
+  })
+_sym_db.RegisterMessage(RemoveAliasRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'environments.environments_pb2'
+  # @@protoc_insertion_point(class_scope:content.environments.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+_ENVIRONMENTS = DESCRIPTOR.services_by_name['Environments']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z<git.perx.ru/perxis/perxis-go/proto/environments;environments'
+  _ENVIRONMENT._serialized_start=87
+  _ENVIRONMENT._serialized_end=358
+  _ENVIRONMENT_STATE._serialized_start=292
+  _ENVIRONMENT_STATE._serialized_end=358
+  _CONFIG._serialized_start=360
+  _CONFIG._serialized_end=405
+  _CREATEREQUEST._serialized_start=407
+  _CREATEREQUEST._serialized_end=470
+  _CREATERESPONSE._serialized_start=472
+  _CREATERESPONSE._serialized_end=540
+  _LISTREQUEST._serialized_start=542
+  _LISTREQUEST._serialized_end=573
+  _LISTRESPONSE._serialized_start=575
+  _LISTRESPONSE._serialized_end=638
+  _GETREQUEST._serialized_start=640
+  _GETREQUEST._serialized_end=686
+  _GETRESPONSE._serialized_start=688
+  _GETRESPONSE._serialized_end=749
+  _UPDATEREQUEST._serialized_start=751
+  _UPDATEREQUEST._serialized_end=814
+  _SETALIASREQUEST._serialized_start=816
+  _SETALIASREQUEST._serialized_end=882
+  _REMOVEALIASREQUEST._serialized_start=884
+  _REMOVEALIASREQUEST._serialized_end=953
+  _DELETEREQUEST._serialized_start=955
+  _DELETEREQUEST._serialized_end=1004
+  _ENVIRONMENTS._serialized_start=1007
+  _ENVIRONMENTS._serialized_end=1573
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/environments/environments_pb2_grpc.py b/clients/environments/environments_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..079945761305b1ec86c05226ae2ea28417113413
--- /dev/null
+++ b/clients/environments/environments_pb2_grpc.py
@@ -0,0 +1,265 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from environments import environments_pb2 as environments_dot_environments__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+class EnvironmentsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.environments.Environments/Create',
+                request_serializer=environments_dot_environments__pb2.CreateRequest.SerializeToString,
+                response_deserializer=environments_dot_environments__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.environments.Environments/Get',
+                request_serializer=environments_dot_environments__pb2.GetRequest.SerializeToString,
+                response_deserializer=environments_dot_environments__pb2.GetResponse.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.environments.Environments/List',
+                request_serializer=environments_dot_environments__pb2.ListRequest.SerializeToString,
+                response_deserializer=environments_dot_environments__pb2.ListResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.environments.Environments/Update',
+                request_serializer=environments_dot_environments__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.SetAlias = channel.unary_unary(
+                '/content.environments.Environments/SetAlias',
+                request_serializer=environments_dot_environments__pb2.SetAliasRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.RemoveAlias = channel.unary_unary(
+                '/content.environments.Environments/RemoveAlias',
+                request_serializer=environments_dot_environments__pb2.RemoveAliasRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.environments.Environments/Delete',
+                request_serializer=environments_dot_environments__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class EnvironmentsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def SetAlias(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def RemoveAlias(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_EnvironmentsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=environments_dot_environments__pb2.CreateRequest.FromString,
+                    response_serializer=environments_dot_environments__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=environments_dot_environments__pb2.GetRequest.FromString,
+                    response_serializer=environments_dot_environments__pb2.GetResponse.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=environments_dot_environments__pb2.ListRequest.FromString,
+                    response_serializer=environments_dot_environments__pb2.ListResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=environments_dot_environments__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'SetAlias': grpc.unary_unary_rpc_method_handler(
+                    servicer.SetAlias,
+                    request_deserializer=environments_dot_environments__pb2.SetAliasRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'RemoveAlias': grpc.unary_unary_rpc_method_handler(
+                    servicer.RemoveAlias,
+                    request_deserializer=environments_dot_environments__pb2.RemoveAliasRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=environments_dot_environments__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.environments.Environments', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Environments(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/Create',
+            environments_dot_environments__pb2.CreateRequest.SerializeToString,
+            environments_dot_environments__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/Get',
+            environments_dot_environments__pb2.GetRequest.SerializeToString,
+            environments_dot_environments__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/List',
+            environments_dot_environments__pb2.ListRequest.SerializeToString,
+            environments_dot_environments__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/Update',
+            environments_dot_environments__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def SetAlias(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/SetAlias',
+            environments_dot_environments__pb2.SetAliasRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def RemoveAlias(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/RemoveAlias',
+            environments_dot_environments__pb2.RemoveAliasRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.environments.Environments/Delete',
+            environments_dot_environments__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/extensions/extension_pb2.py b/clients/extensions/extension_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..a9996aade1aa994dbc8193db1c7c42376932d141
--- /dev/null
+++ b/clients/extensions/extension_pb2.py
@@ -0,0 +1,184 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: extensions/extension.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from references import references_pb2 as references_dot_references__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1a\x65xtensions/extension.proto\x12\nextensions\x1a\x1breferences/references.proto\"Y\n\x0eInstallRequest\x12\x13\n\nextensions\x18\x90N \x03(\t\x12\x11\n\x08space_id\x18\x9aN \x01(\t\x12\x0f\n\x06\x65nv_id\x18\xa4N \x01(\t\x12\x0e\n\x05\x66orce\x18\xf4N \x01(\x08\"\xd9\x01\n\x16\x45xtensionRequestResult\x12\x38\n\x05state\x18\x90N \x01(\x0e\x32(.extensions.ExtensionRequestResult.State\x12\x12\n\textension\x18\xf4N \x01(\t\x12\x0c\n\x03msg\x18\xd8O \x01(\t\x12\x0e\n\x05\x65rror\x18\xbcP \x01(\t\x12\x19\n\x10update_available\x18\xa0Q \x01(\x08\"8\n\x05State\x12\x06\n\x02OK\x10\x00\x12\t\n\x05\x45RROR\x10\x01\x12\x0b\n\x07PENDING\x10\x02\x12\x0f\n\x0bIN_PROGRESS\x10\x03\"G\n\x0fInstallResponse\x12\x34\n\x07results\x18\x90N \x03(\x0b\x32\".extensions.ExtensionRequestResult\"l\n\x10UninstallRequest\x12\x13\n\nextensions\x18\x90N \x03(\t\x12\x11\n\x08space_id\x18\x9aN \x01(\t\x12\x0f\n\x06\x65nv_id\x18\xa4N \x01(\t\x12\x0f\n\x06remove\x18\xf4N \x01(\x08\x12\x0e\n\x05\x66orce\x18\xd8O \x01(\x08\"I\n\x11UninstallResponse\x12\x34\n\x07results\x18\x90N \x03(\x0b\x32\".extensions.ExtensionRequestResult\"X\n\rUpdateRequest\x12\x13\n\nextensions\x18\x90N \x03(\t\x12\x11\n\x08space_id\x18\x9aN \x01(\t\x12\x0f\n\x06\x65nv_id\x18\xa4N \x01(\t\x12\x0e\n\x05\x66orce\x18\xf4N \x01(\x08\"F\n\x0eUpdateResponse\x12\x34\n\x07results\x18\x90N \x03(\x0b\x32\".extensions.ExtensionRequestResult\"G\n\x0c\x43heckRequest\x12\x13\n\nextensions\x18\x90N \x03(\t\x12\x11\n\x08space_id\x18\x9aN \x01(\t\x12\x0f\n\x06\x65nv_id\x18\xa4N \x01(\t\"E\n\rCheckResponse\x12\x34\n\x07results\x18\x90N \x03(\x0b\x32\".extensions.ExtensionRequestResult\"\xf1\x02\n\rActionRequest\x12\x12\n\textension\x18\xe8\x07 \x01(\t\x12\x0f\n\x06\x61\x63tion\x18\xf4N \x01(\t\x12\x11\n\x08space_id\x18\x84R \x01(\t\x12\x0f\n\x06\x65nv_id\x18\x8eR \x01(\t\x12\x16\n\rcollection_id\x18\x98R \x01(\t\x12\x10\n\x07item_id\x18\xa2R \x01(\t\x12\x11\n\x08item_ids\x18\xacR \x03(\t\x12\x0f\n\x06\x66ields\x18\xb6R \x03(\t\x12:\n\x08metadata\x18\xf8U \x03(\x0b\x32\'.extensions.ActionRequest.MetadataEntry\x12,\n\x04refs\x18\x82V \x03(\x0b\x32\x1d.content.references.Reference\x12.\n\x06params\x18\x8cV \x01(\x0b\x32\x1d.content.references.Reference\x1a/\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\"\xdb\x02\n\x0e\x41\x63tionResponse\x12\x30\n\x05state\x18\x90N \x01(\x0e\x32 .extensions.ActionResponse.State\x12\x0c\n\x03msg\x18\xf4N \x01(\t\x12\x0e\n\x05\x65rror\x18\xd8O \x01(\t\x12!\n\x04next\x18\xbcP \x03(\x0b\x32\x12.extensions.Action\x12;\n\x08metadata\x18\xa0Q \x03(\x0b\x32(.extensions.ActionResponse.MetadataEntry\x12,\n\x04refs\x18\xd0P \x03(\x0b\x32\x1d.content.references.Reference\x1a/\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\":\n\x05State\x12\x08\n\x04\x44ONE\x10\x00\x12\t\n\x05\x45RROR\x10\x01\x12\x0b\n\x07PENDING\x10\x02\x12\x0f\n\x0bIN_PROGRESS\x10\x03\"\xf9\x03\n\x06\x41\x63tion\x12\x12\n\textension\x18\x90N \x01(\t\x12\x0f\n\x06\x61\x63tion\x18\xf4N \x01(\t\x12\r\n\x04name\x18\xd8O \x01(\t\x12\x14\n\x0b\x64\x65scription\x18\xe2O \x01(\t\x12\r\n\x04icon\x18\xecO \x01(\t\x12-\n\x05image\x18\xf6O \x01(\x0b\x32\x1d.content.references.Reference\x12\x0f\n\x06groups\x18\x80P \x03(\t\x12&\n\x04kind\x18\xbcP \x01(\x0e\x32\x17.extensions.Action.Kind\x12\x10\n\x07\x63lasses\x18\xc6P \x03(\t\x12,\n\x04refs\x18\xd0P \x03(\x0b\x32\x1d.content.references.Reference\x12\x1a\n\x11params_collection\x18\xdaP \x01(\t\x12+\n\x07request\x18\xa0Q \x01(\x0b\x32\x19.extensions.ActionRequest\x12\x1a\n\x11navigation_action\x18\x84R \x01(\x08\x12\x19\n\x10navigation_route\x18\x8eR \x01(\t\"n\n\x04Kind\x12\x0b\n\x07\x44\x45\x46\x41ULT\x10\x00\x12\t\n\x05SPACE\x10\x01\x12\x0f\n\x0b\x45NVIRONMENT\x10\x02\x12\x0e\n\nCOLLECTION\x10\x03\x12\x08\n\x04ITEM\x10\x04\x12\t\n\x05ITEMS\x10\x05\x12\x0c\n\x08REVISION\x10\x06\x12\n\n\x06\x43REATE\x10\x07\x32\xe3\x02\n\tExtension\x12\x44\n\x07Install\x12\x1a.extensions.InstallRequest\x1a\x1b.extensions.InstallResponse\"\x00\x12>\n\x05\x43heck\x12\x18.extensions.CheckRequest\x1a\x19.extensions.CheckResponse\"\x00\x12\x41\n\x06Update\x12\x19.extensions.UpdateRequest\x1a\x1a.extensions.UpdateResponse\"\x00\x12J\n\tUninstall\x12\x1c.extensions.UninstallRequest\x1a\x1d.extensions.UninstallResponse\"\x00\x12\x41\n\x06\x41\x63tion\x12\x19.extensions.ActionRequest\x1a\x1a.extensions.ActionResponse\"\x00\x42:Z8git.perx.ru/perxis/perxis-go/proto/extensions;extensionsb\x06proto3')
+
+
+
+_INSTALLREQUEST = DESCRIPTOR.message_types_by_name['InstallRequest']
+_EXTENSIONREQUESTRESULT = DESCRIPTOR.message_types_by_name['ExtensionRequestResult']
+_INSTALLRESPONSE = DESCRIPTOR.message_types_by_name['InstallResponse']
+_UNINSTALLREQUEST = DESCRIPTOR.message_types_by_name['UninstallRequest']
+_UNINSTALLRESPONSE = DESCRIPTOR.message_types_by_name['UninstallResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_UPDATERESPONSE = DESCRIPTOR.message_types_by_name['UpdateResponse']
+_CHECKREQUEST = DESCRIPTOR.message_types_by_name['CheckRequest']
+_CHECKRESPONSE = DESCRIPTOR.message_types_by_name['CheckResponse']
+_ACTIONREQUEST = DESCRIPTOR.message_types_by_name['ActionRequest']
+_ACTIONREQUEST_METADATAENTRY = _ACTIONREQUEST.nested_types_by_name['MetadataEntry']
+_ACTIONRESPONSE = DESCRIPTOR.message_types_by_name['ActionResponse']
+_ACTIONRESPONSE_METADATAENTRY = _ACTIONRESPONSE.nested_types_by_name['MetadataEntry']
+_ACTION = DESCRIPTOR.message_types_by_name['Action']
+_EXTENSIONREQUESTRESULT_STATE = _EXTENSIONREQUESTRESULT.enum_types_by_name['State']
+_ACTIONRESPONSE_STATE = _ACTIONRESPONSE.enum_types_by_name['State']
+_ACTION_KIND = _ACTION.enum_types_by_name['Kind']
+InstallRequest = _reflection.GeneratedProtocolMessageType('InstallRequest', (_message.Message,), {
+  'DESCRIPTOR' : _INSTALLREQUEST,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.InstallRequest)
+  })
+_sym_db.RegisterMessage(InstallRequest)
+
+ExtensionRequestResult = _reflection.GeneratedProtocolMessageType('ExtensionRequestResult', (_message.Message,), {
+  'DESCRIPTOR' : _EXTENSIONREQUESTRESULT,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ExtensionRequestResult)
+  })
+_sym_db.RegisterMessage(ExtensionRequestResult)
+
+InstallResponse = _reflection.GeneratedProtocolMessageType('InstallResponse', (_message.Message,), {
+  'DESCRIPTOR' : _INSTALLRESPONSE,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.InstallResponse)
+  })
+_sym_db.RegisterMessage(InstallResponse)
+
+UninstallRequest = _reflection.GeneratedProtocolMessageType('UninstallRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UNINSTALLREQUEST,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UninstallRequest)
+  })
+_sym_db.RegisterMessage(UninstallRequest)
+
+UninstallResponse = _reflection.GeneratedProtocolMessageType('UninstallResponse', (_message.Message,), {
+  'DESCRIPTOR' : _UNINSTALLRESPONSE,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UninstallResponse)
+  })
+_sym_db.RegisterMessage(UninstallResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+UpdateResponse = _reflection.GeneratedProtocolMessageType('UpdateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATERESPONSE,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UpdateResponse)
+  })
+_sym_db.RegisterMessage(UpdateResponse)
+
+CheckRequest = _reflection.GeneratedProtocolMessageType('CheckRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CHECKREQUEST,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.CheckRequest)
+  })
+_sym_db.RegisterMessage(CheckRequest)
+
+CheckResponse = _reflection.GeneratedProtocolMessageType('CheckResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CHECKRESPONSE,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.CheckResponse)
+  })
+_sym_db.RegisterMessage(CheckResponse)
+
+ActionRequest = _reflection.GeneratedProtocolMessageType('ActionRequest', (_message.Message,), {
+
+  'MetadataEntry' : _reflection.GeneratedProtocolMessageType('MetadataEntry', (_message.Message,), {
+    'DESCRIPTOR' : _ACTIONREQUEST_METADATAENTRY,
+    '__module__' : 'extensions.extension_pb2'
+    # @@protoc_insertion_point(class_scope:extensions.ActionRequest.MetadataEntry)
+    })
+  ,
+  'DESCRIPTOR' : _ACTIONREQUEST,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ActionRequest)
+  })
+_sym_db.RegisterMessage(ActionRequest)
+_sym_db.RegisterMessage(ActionRequest.MetadataEntry)
+
+ActionResponse = _reflection.GeneratedProtocolMessageType('ActionResponse', (_message.Message,), {
+
+  'MetadataEntry' : _reflection.GeneratedProtocolMessageType('MetadataEntry', (_message.Message,), {
+    'DESCRIPTOR' : _ACTIONRESPONSE_METADATAENTRY,
+    '__module__' : 'extensions.extension_pb2'
+    # @@protoc_insertion_point(class_scope:extensions.ActionResponse.MetadataEntry)
+    })
+  ,
+  'DESCRIPTOR' : _ACTIONRESPONSE,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ActionResponse)
+  })
+_sym_db.RegisterMessage(ActionResponse)
+_sym_db.RegisterMessage(ActionResponse.MetadataEntry)
+
+Action = _reflection.GeneratedProtocolMessageType('Action', (_message.Message,), {
+  'DESCRIPTOR' : _ACTION,
+  '__module__' : 'extensions.extension_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.Action)
+  })
+_sym_db.RegisterMessage(Action)
+
+_EXTENSION = DESCRIPTOR.services_by_name['Extension']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z8git.perx.ru/perxis/perxis-go/proto/extensions;extensions'
+  _ACTIONREQUEST_METADATAENTRY._options = None
+  _ACTIONREQUEST_METADATAENTRY._serialized_options = b'8\001'
+  _ACTIONRESPONSE_METADATAENTRY._options = None
+  _ACTIONRESPONSE_METADATAENTRY._serialized_options = b'8\001'
+  _INSTALLREQUEST._serialized_start=71
+  _INSTALLREQUEST._serialized_end=160
+  _EXTENSIONREQUESTRESULT._serialized_start=163
+  _EXTENSIONREQUESTRESULT._serialized_end=380
+  _EXTENSIONREQUESTRESULT_STATE._serialized_start=324
+  _EXTENSIONREQUESTRESULT_STATE._serialized_end=380
+  _INSTALLRESPONSE._serialized_start=382
+  _INSTALLRESPONSE._serialized_end=453
+  _UNINSTALLREQUEST._serialized_start=455
+  _UNINSTALLREQUEST._serialized_end=563
+  _UNINSTALLRESPONSE._serialized_start=565
+  _UNINSTALLRESPONSE._serialized_end=638
+  _UPDATEREQUEST._serialized_start=640
+  _UPDATEREQUEST._serialized_end=728
+  _UPDATERESPONSE._serialized_start=730
+  _UPDATERESPONSE._serialized_end=800
+  _CHECKREQUEST._serialized_start=802
+  _CHECKREQUEST._serialized_end=873
+  _CHECKRESPONSE._serialized_start=875
+  _CHECKRESPONSE._serialized_end=944
+  _ACTIONREQUEST._serialized_start=947
+  _ACTIONREQUEST._serialized_end=1316
+  _ACTIONREQUEST_METADATAENTRY._serialized_start=1269
+  _ACTIONREQUEST_METADATAENTRY._serialized_end=1316
+  _ACTIONRESPONSE._serialized_start=1319
+  _ACTIONRESPONSE._serialized_end=1666
+  _ACTIONRESPONSE_METADATAENTRY._serialized_start=1269
+  _ACTIONRESPONSE_METADATAENTRY._serialized_end=1316
+  _ACTIONRESPONSE_STATE._serialized_start=1608
+  _ACTIONRESPONSE_STATE._serialized_end=1666
+  _ACTION._serialized_start=1669
+  _ACTION._serialized_end=2174
+  _ACTION_KIND._serialized_start=2064
+  _ACTION_KIND._serialized_end=2174
+  _EXTENSION._serialized_start=2177
+  _EXTENSION._serialized_end=2532
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/extensions/extension_pb2_grpc.py b/clients/extensions/extension_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..e1251c165622381f23453434522c731995480642
--- /dev/null
+++ b/clients/extensions/extension_pb2_grpc.py
@@ -0,0 +1,227 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from extensions import extension_pb2 as extensions_dot_extension__pb2
+
+
+class ExtensionStub(object):
+    """Extension - API расширения
+    """
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Install = channel.unary_unary(
+                '/extensions.Extension/Install',
+                request_serializer=extensions_dot_extension__pb2.InstallRequest.SerializeToString,
+                response_deserializer=extensions_dot_extension__pb2.InstallResponse.FromString,
+                )
+        self.Check = channel.unary_unary(
+                '/extensions.Extension/Check',
+                request_serializer=extensions_dot_extension__pb2.CheckRequest.SerializeToString,
+                response_deserializer=extensions_dot_extension__pb2.CheckResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/extensions.Extension/Update',
+                request_serializer=extensions_dot_extension__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=extensions_dot_extension__pb2.UpdateResponse.FromString,
+                )
+        self.Uninstall = channel.unary_unary(
+                '/extensions.Extension/Uninstall',
+                request_serializer=extensions_dot_extension__pb2.UninstallRequest.SerializeToString,
+                response_deserializer=extensions_dot_extension__pb2.UninstallResponse.FromString,
+                )
+        self.Action = channel.unary_unary(
+                '/extensions.Extension/Action',
+                request_serializer=extensions_dot_extension__pb2.ActionRequest.SerializeToString,
+                response_deserializer=extensions_dot_extension__pb2.ActionResponse.FromString,
+                )
+
+
+class ExtensionServicer(object):
+    """Extension - API расширения
+    """
+
+    def Install(self, request, context):
+        """Для установки расширения выполняется запрос к сервису с указанием расширений которые должны быть установлены. Если
+        расширение уже установлено процесс возвращает ошибку.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Check(self, request, context):
+        """Можно запросить проверку статуса установки для расширения. При этом расширение проверяет наличие необходимых данных в
+        пространстве или наличие новой версии расширения и сообщает об этом. Никаких действий с данными пространства не
+        производится.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Сервис выполняет необходимые действия с данными и миграции для соответствия нужной версии расширения.
+        Если расширение не установлено процесс обновления возвращает ошибку.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Uninstall(self, request, context):
+        """Удаление расширения из пространства.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Action(self, request, context):
+        """Пользовательские Действия (Actions)
+
+        Пользовательские действия позволяют расширить функционал пользовательского интерфейса путем
+        добавления в интерфейс элементов взаимодействуя с которыми пользователь может вызывать реакцию на
+        сервере или переход в интерфейсе.
+
+        Пользовательские действия добавляются при установке расширений в системную коллекции `System/Actions`.
+        Коллекция создается автоматически менеджером расширений. При установке так же отображается меню `Действия`
+        доступное для всех пользователей.
+
+        Примеры пользовательских действий:
+        - "Собрать сайт" - добавляется Perxis.Web для сборки сайта, доступна пользователю через меню,
+        параметры space_id, env_id. При вызове выполняется запрос на сервер ProcessAction.
+        - "Посмотреть задачи" - добавляется Tasks для перехода на коллекцию задач. Отображается в меню,
+        параметры space_id, env_id.
+
+        Приложения так же могут использовать действия для вызова обработки в других приложениях при
+        необходимости.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ExtensionServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Install': grpc.unary_unary_rpc_method_handler(
+                    servicer.Install,
+                    request_deserializer=extensions_dot_extension__pb2.InstallRequest.FromString,
+                    response_serializer=extensions_dot_extension__pb2.InstallResponse.SerializeToString,
+            ),
+            'Check': grpc.unary_unary_rpc_method_handler(
+                    servicer.Check,
+                    request_deserializer=extensions_dot_extension__pb2.CheckRequest.FromString,
+                    response_serializer=extensions_dot_extension__pb2.CheckResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=extensions_dot_extension__pb2.UpdateRequest.FromString,
+                    response_serializer=extensions_dot_extension__pb2.UpdateResponse.SerializeToString,
+            ),
+            'Uninstall': grpc.unary_unary_rpc_method_handler(
+                    servicer.Uninstall,
+                    request_deserializer=extensions_dot_extension__pb2.UninstallRequest.FromString,
+                    response_serializer=extensions_dot_extension__pb2.UninstallResponse.SerializeToString,
+            ),
+            'Action': grpc.unary_unary_rpc_method_handler(
+                    servicer.Action,
+                    request_deserializer=extensions_dot_extension__pb2.ActionRequest.FromString,
+                    response_serializer=extensions_dot_extension__pb2.ActionResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'extensions.Extension', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Extension(object):
+    """Extension - API расширения
+    """
+
+    @staticmethod
+    def Install(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.Extension/Install',
+            extensions_dot_extension__pb2.InstallRequest.SerializeToString,
+            extensions_dot_extension__pb2.InstallResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Check(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.Extension/Check',
+            extensions_dot_extension__pb2.CheckRequest.SerializeToString,
+            extensions_dot_extension__pb2.CheckResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.Extension/Update',
+            extensions_dot_extension__pb2.UpdateRequest.SerializeToString,
+            extensions_dot_extension__pb2.UpdateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Uninstall(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.Extension/Uninstall',
+            extensions_dot_extension__pb2.UninstallRequest.SerializeToString,
+            extensions_dot_extension__pb2.UninstallResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Action(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.Extension/Action',
+            extensions_dot_extension__pb2.ActionRequest.SerializeToString,
+            extensions_dot_extension__pb2.ActionResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/extensions/manager_pb2.py b/clients/extensions/manager_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..ca25fdafd7179d7789f50c7b4f26fd09275cedd0
--- /dev/null
+++ b/clients/extensions/manager_pb2.py
@@ -0,0 +1,134 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: extensions/manager.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x18\x65xtensions/manager.proto\x12\nextensions\"\x8d\x02\n\x13\x45xtensionDescriptor\x12\x12\n\textension\x18\x90N \x01(\t\x12\x0e\n\x05title\x18\x9aN \x01(\t\x12\x14\n\x0b\x64\x65scription\x18\xa4N \x01(\t\x12\x10\n\x07version\x18\xf4N \x01(\t\x12\x1c\n\x13version_description\x18\xfeN \x01(\t\x12\r\n\x04\x64\x65ps\x18\xd8O \x03(\t\x12\x0b\n\x03url\x18\x05 \x01(\t\x12?\n\x08metadata\x18\x06 \x03(\x0b\x32-.extensions.ExtensionDescriptor.MetadataEntry\x1a/\n\rMetadataEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x01(\t:\x02\x38\x01\"P\n\x19RegisterExtensionsRequest\x12\x33\n\nextensions\x18\x01 \x03(\x0b\x32\x1f.extensions.ExtensionDescriptor\"\x1c\n\x1aRegisterExtensionsResponse\"R\n\x1bUnregisterExtensionsRequest\x12\x33\n\nextensions\x18\x01 \x03(\x0b\x32\x1f.extensions.ExtensionDescriptor\"\x1e\n\x1cUnregisterExtensionsResponse\")\n\x14ListExtensionsFilter\x12\x11\n\textension\x18\x01 \x03(\t\"I\n\x15ListExtensionsRequest\x12\x30\n\x06\x66ilter\x18\x01 \x01(\x0b\x32 .extensions.ListExtensionsFilter\"M\n\x16ListExtensionsResponse\x12\x33\n\nextensions\x18\x01 \x03(\x0b\x32\x1f.extensions.ExtensionDescriptor\"\xf5\x01\n\x0fSpaceExtensions\x12\x12\n\textension\x18\x90N \x01(\t\x12\x0e\n\x05title\x18\x9aN \x01(\t\x12\x10\n\x07version\x18\xf4N \x01(\t\x12\r\n\x04\x64\x65ps\x18\xd8O \x03(\t\x12\x31\n\x05state\x18\xbcP \x01(\x0e\x32!.extensions.SpaceExtensions.State\x12\x15\n\x0cstatus_error\x18\xa0Q \x01(\t\x12\x13\n\nstatus_msg\x18\x84R \x01(\t\">\n\x05State\x12\x0b\n\x07PENDING\x10\x00\x12\r\n\tINSTALLED\x10\x01\x12\x0f\n\x0bIN_PROGRESS\x10\x02\x12\x08\n\x04\x46\x41IL\x10\x03\x32\xc1\x02\n\x10\x45xtensionManager\x12\x65\n\x12RegisterExtensions\x12%.extensions.RegisterExtensionsRequest\x1a&.extensions.RegisterExtensionsResponse\"\x00\x12k\n\x14UnregisterExtensions\x12\'.extensions.UnregisterExtensionsRequest\x1a(.extensions.UnregisterExtensionsResponse\"\x00\x12Y\n\x0eListExtensions\x12!.extensions.ListExtensionsRequest\x1a\".extensions.ListExtensionsResponse\"\x00\x42:Z8git.perx.ru/perxis/perxis-go/proto/extensions;extensionsb\x06proto3')
+
+
+
+_EXTENSIONDESCRIPTOR = DESCRIPTOR.message_types_by_name['ExtensionDescriptor']
+_EXTENSIONDESCRIPTOR_METADATAENTRY = _EXTENSIONDESCRIPTOR.nested_types_by_name['MetadataEntry']
+_REGISTEREXTENSIONSREQUEST = DESCRIPTOR.message_types_by_name['RegisterExtensionsRequest']
+_REGISTEREXTENSIONSRESPONSE = DESCRIPTOR.message_types_by_name['RegisterExtensionsResponse']
+_UNREGISTEREXTENSIONSREQUEST = DESCRIPTOR.message_types_by_name['UnregisterExtensionsRequest']
+_UNREGISTEREXTENSIONSRESPONSE = DESCRIPTOR.message_types_by_name['UnregisterExtensionsResponse']
+_LISTEXTENSIONSFILTER = DESCRIPTOR.message_types_by_name['ListExtensionsFilter']
+_LISTEXTENSIONSREQUEST = DESCRIPTOR.message_types_by_name['ListExtensionsRequest']
+_LISTEXTENSIONSRESPONSE = DESCRIPTOR.message_types_by_name['ListExtensionsResponse']
+_SPACEEXTENSIONS = DESCRIPTOR.message_types_by_name['SpaceExtensions']
+_SPACEEXTENSIONS_STATE = _SPACEEXTENSIONS.enum_types_by_name['State']
+ExtensionDescriptor = _reflection.GeneratedProtocolMessageType('ExtensionDescriptor', (_message.Message,), {
+
+  'MetadataEntry' : _reflection.GeneratedProtocolMessageType('MetadataEntry', (_message.Message,), {
+    'DESCRIPTOR' : _EXTENSIONDESCRIPTOR_METADATAENTRY,
+    '__module__' : 'extensions.manager_pb2'
+    # @@protoc_insertion_point(class_scope:extensions.ExtensionDescriptor.MetadataEntry)
+    })
+  ,
+  'DESCRIPTOR' : _EXTENSIONDESCRIPTOR,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ExtensionDescriptor)
+  })
+_sym_db.RegisterMessage(ExtensionDescriptor)
+_sym_db.RegisterMessage(ExtensionDescriptor.MetadataEntry)
+
+RegisterExtensionsRequest = _reflection.GeneratedProtocolMessageType('RegisterExtensionsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _REGISTEREXTENSIONSREQUEST,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.RegisterExtensionsRequest)
+  })
+_sym_db.RegisterMessage(RegisterExtensionsRequest)
+
+RegisterExtensionsResponse = _reflection.GeneratedProtocolMessageType('RegisterExtensionsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _REGISTEREXTENSIONSRESPONSE,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.RegisterExtensionsResponse)
+  })
+_sym_db.RegisterMessage(RegisterExtensionsResponse)
+
+UnregisterExtensionsRequest = _reflection.GeneratedProtocolMessageType('UnregisterExtensionsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UNREGISTEREXTENSIONSREQUEST,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UnregisterExtensionsRequest)
+  })
+_sym_db.RegisterMessage(UnregisterExtensionsRequest)
+
+UnregisterExtensionsResponse = _reflection.GeneratedProtocolMessageType('UnregisterExtensionsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _UNREGISTEREXTENSIONSRESPONSE,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.UnregisterExtensionsResponse)
+  })
+_sym_db.RegisterMessage(UnregisterExtensionsResponse)
+
+ListExtensionsFilter = _reflection.GeneratedProtocolMessageType('ListExtensionsFilter', (_message.Message,), {
+  'DESCRIPTOR' : _LISTEXTENSIONSFILTER,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ListExtensionsFilter)
+  })
+_sym_db.RegisterMessage(ListExtensionsFilter)
+
+ListExtensionsRequest = _reflection.GeneratedProtocolMessageType('ListExtensionsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTEXTENSIONSREQUEST,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ListExtensionsRequest)
+  })
+_sym_db.RegisterMessage(ListExtensionsRequest)
+
+ListExtensionsResponse = _reflection.GeneratedProtocolMessageType('ListExtensionsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTEXTENSIONSRESPONSE,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.ListExtensionsResponse)
+  })
+_sym_db.RegisterMessage(ListExtensionsResponse)
+
+SpaceExtensions = _reflection.GeneratedProtocolMessageType('SpaceExtensions', (_message.Message,), {
+  'DESCRIPTOR' : _SPACEEXTENSIONS,
+  '__module__' : 'extensions.manager_pb2'
+  # @@protoc_insertion_point(class_scope:extensions.SpaceExtensions)
+  })
+_sym_db.RegisterMessage(SpaceExtensions)
+
+_EXTENSIONMANAGER = DESCRIPTOR.services_by_name['ExtensionManager']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z8git.perx.ru/perxis/perxis-go/proto/extensions;extensions'
+  _EXTENSIONDESCRIPTOR_METADATAENTRY._options = None
+  _EXTENSIONDESCRIPTOR_METADATAENTRY._serialized_options = b'8\001'
+  _EXTENSIONDESCRIPTOR._serialized_start=41
+  _EXTENSIONDESCRIPTOR._serialized_end=310
+  _EXTENSIONDESCRIPTOR_METADATAENTRY._serialized_start=263
+  _EXTENSIONDESCRIPTOR_METADATAENTRY._serialized_end=310
+  _REGISTEREXTENSIONSREQUEST._serialized_start=312
+  _REGISTEREXTENSIONSREQUEST._serialized_end=392
+  _REGISTEREXTENSIONSRESPONSE._serialized_start=394
+  _REGISTEREXTENSIONSRESPONSE._serialized_end=422
+  _UNREGISTEREXTENSIONSREQUEST._serialized_start=424
+  _UNREGISTEREXTENSIONSREQUEST._serialized_end=506
+  _UNREGISTEREXTENSIONSRESPONSE._serialized_start=508
+  _UNREGISTEREXTENSIONSRESPONSE._serialized_end=538
+  _LISTEXTENSIONSFILTER._serialized_start=540
+  _LISTEXTENSIONSFILTER._serialized_end=581
+  _LISTEXTENSIONSREQUEST._serialized_start=583
+  _LISTEXTENSIONSREQUEST._serialized_end=656
+  _LISTEXTENSIONSRESPONSE._serialized_start=658
+  _LISTEXTENSIONSRESPONSE._serialized_end=735
+  _SPACEEXTENSIONS._serialized_start=738
+  _SPACEEXTENSIONS._serialized_end=983
+  _SPACEEXTENSIONS_STATE._serialized_start=921
+  _SPACEEXTENSIONS_STATE._serialized_end=983
+  _EXTENSIONMANAGER._serialized_start=986
+  _EXTENSIONMANAGER._serialized_end=1307
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/extensions/manager_pb2_grpc.py b/clients/extensions/manager_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..8fad6c1e2bb4cde1acc1eed54dfdb40b1c1fd4d7
--- /dev/null
+++ b/clients/extensions/manager_pb2_grpc.py
@@ -0,0 +1,148 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from extensions import manager_pb2 as extensions_dot_manager__pb2
+
+
+class ExtensionManagerStub(object):
+    """ExtensionManager - менеджер расширений. Должен реализовывать так же сервис Extension
+    """
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.RegisterExtensions = channel.unary_unary(
+                '/extensions.ExtensionManager/RegisterExtensions',
+                request_serializer=extensions_dot_manager__pb2.RegisterExtensionsRequest.SerializeToString,
+                response_deserializer=extensions_dot_manager__pb2.RegisterExtensionsResponse.FromString,
+                )
+        self.UnregisterExtensions = channel.unary_unary(
+                '/extensions.ExtensionManager/UnregisterExtensions',
+                request_serializer=extensions_dot_manager__pb2.UnregisterExtensionsRequest.SerializeToString,
+                response_deserializer=extensions_dot_manager__pb2.UnregisterExtensionsResponse.FromString,
+                )
+        self.ListExtensions = channel.unary_unary(
+                '/extensions.ExtensionManager/ListExtensions',
+                request_serializer=extensions_dot_manager__pb2.ListExtensionsRequest.SerializeToString,
+                response_deserializer=extensions_dot_manager__pb2.ListExtensionsResponse.FromString,
+                )
+
+
+class ExtensionManagerServicer(object):
+    """ExtensionManager - менеджер расширений. Должен реализовывать так же сервис Extension
+    """
+
+    def RegisterExtensions(self, request, context):
+        """##  Регистрация расширений
+
+        Регистрация\Дерегистрация происходить через сервис менеджера расширений (Extension Manager). В процессе регистрации сервис сообщает
+        о поддерживаемых сервисом действиях (Actions) и версии сервиса и зависимостях:
+        1. Действия (Actions) - перечень действия которые обрабатываются сервисом. Включает в себя как системные действия,
+        так и пользовательские, которые могут быть использованы в интерфейсе;
+        2. Имя сервиса/расширения - название сервиса внутри системы;
+        3. Версия сервиса - сервис сообщает текущую версию сервиса. Контроллер сообщает пользователю о возможности обновления
+        расширения при смене версии;
+        4. Описание версии - содержит информацию об изменениях в последних версиях расширения;
+        5. Зависимости - перечень расширений которые необходимы сервису для функционирования. При установке сервиса
+        все расширения от которых он зависит, будут так же установлены.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def UnregisterExtensions(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListExtensions(self, request, context):
+        """Получить список зарегистрированных сервисов
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ExtensionManagerServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'RegisterExtensions': grpc.unary_unary_rpc_method_handler(
+                    servicer.RegisterExtensions,
+                    request_deserializer=extensions_dot_manager__pb2.RegisterExtensionsRequest.FromString,
+                    response_serializer=extensions_dot_manager__pb2.RegisterExtensionsResponse.SerializeToString,
+            ),
+            'UnregisterExtensions': grpc.unary_unary_rpc_method_handler(
+                    servicer.UnregisterExtensions,
+                    request_deserializer=extensions_dot_manager__pb2.UnregisterExtensionsRequest.FromString,
+                    response_serializer=extensions_dot_manager__pb2.UnregisterExtensionsResponse.SerializeToString,
+            ),
+            'ListExtensions': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListExtensions,
+                    request_deserializer=extensions_dot_manager__pb2.ListExtensionsRequest.FromString,
+                    response_serializer=extensions_dot_manager__pb2.ListExtensionsResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'extensions.ExtensionManager', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class ExtensionManager(object):
+    """ExtensionManager - менеджер расширений. Должен реализовывать так же сервис Extension
+    """
+
+    @staticmethod
+    def RegisterExtensions(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.ExtensionManager/RegisterExtensions',
+            extensions_dot_manager__pb2.RegisterExtensionsRequest.SerializeToString,
+            extensions_dot_manager__pb2.RegisterExtensionsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def UnregisterExtensions(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.ExtensionManager/UnregisterExtensions',
+            extensions_dot_manager__pb2.UnregisterExtensionsRequest.SerializeToString,
+            extensions_dot_manager__pb2.UnregisterExtensionsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListExtensions(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/extensions.ExtensionManager/ListExtensions',
+            extensions_dot_manager__pb2.ListExtensionsRequest.SerializeToString,
+            extensions_dot_manager__pb2.ListExtensionsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/files/files_pb2.py b/clients/files/files_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..28ecd0a5b2ffc76dfd1b2ac9cca869b1154eddb2
--- /dev/null
+++ b/clients/files/files_pb2.py
@@ -0,0 +1,209 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: files/files.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x11\x66iles/files.proto\x12\x05\x66iles\x1a\x1bgoogle/protobuf/empty.proto\"N\n\x04\x46ile\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x0c\n\x04size\x18\x03 \x01(\x05\x12\x11\n\tmime_type\x18\x04 \x01(\t\x12\x0b\n\x03url\x18\x05 \x01(\t\"\x8a\x01\n\x0fMultipartUpload\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\x12\x11\n\tupload_id\x18\x02 \x01(\t\x12\x11\n\tpart_size\x18\x03 \x01(\x05\x12\x11\n\tpart_urls\x18\x04 \x03(\t\x12#\n\x05parts\x18\x05 \x03(\x0b\x32\x14.files.CompletedPart\"7\n\x06Upload\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\x12\x12\n\nupload_url\x18\x02 \x01(\t\"+\n\rCompletedPart\x12\x0e\n\x06number\x18\x01 \x01(\x05\x12\n\n\x02id\x18\x02 \x01(\t\"<\n\x12StartUploadRequest\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"=\n\x13StartUploadResponse\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"?\n\x15\x43ompleteUploadRequest\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"@\n\x16\x43ompleteUploadResponse\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"<\n\x12\x41\x62ortUploadRequest\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"\x15\n\x13\x41\x62ortUploadResponse\";\n\x11MoveUploadRequest\x12&\n\x06upload\x18\x01 \x01(\x0b\x32\x16.files.MultipartUpload\"/\n\x12MoveUploadResponse\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\"*\n\rUploadRequest\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\"/\n\x0eUploadResponse\x12\x1d\n\x06upload\x18\x01 \x01(\x0b\x32\r.files.Upload\"\x1c\n\x0eGetFileRequest\x12\n\n\x02id\x18\x01 \x01(\t\",\n\x0fGetFileResponse\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\".\n\x11\x44\x65leteFileRequest\x12\x19\n\x04\x66ile\x18\x01 \x01(\x0b\x32\x0b.files.File\"\x14\n\x12\x44\x65leteFileResponse2\xe0\x03\n\x05\x46iles\x12\x46\n\x0bStartUpload\x12\x19.files.StartUploadRequest\x1a\x1a.files.StartUploadResponse\"\x00\x12O\n\x0e\x43ompleteUpload\x12\x1c.files.CompleteUploadRequest\x1a\x1d.files.CompleteUploadResponse\"\x00\x12\x42\n\x0b\x41\x62ortUpload\x12\x19.files.AbortUploadRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x43\n\nMoveUpload\x12\x18.files.MoveUploadRequest\x1a\x19.files.MoveUploadResponse\"\x00\x12\x37\n\x06Upload\x12\x14.files.UploadRequest\x1a\x15.files.UploadResponse\"\x00\x12:\n\x07GetFile\x12\x15.files.GetFileRequest\x1a\x16.files.GetFileResponse\"\x00\x12@\n\nDeleteFile\x12\x18.files.DeleteFileRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x30Z.git.perx.ru/perxis/perxis-go/proto/files;filesb\x06proto3')
+
+
+
+_FILE = DESCRIPTOR.message_types_by_name['File']
+_MULTIPARTUPLOAD = DESCRIPTOR.message_types_by_name['MultipartUpload']
+_UPLOAD = DESCRIPTOR.message_types_by_name['Upload']
+_COMPLETEDPART = DESCRIPTOR.message_types_by_name['CompletedPart']
+_STARTUPLOADREQUEST = DESCRIPTOR.message_types_by_name['StartUploadRequest']
+_STARTUPLOADRESPONSE = DESCRIPTOR.message_types_by_name['StartUploadResponse']
+_COMPLETEUPLOADREQUEST = DESCRIPTOR.message_types_by_name['CompleteUploadRequest']
+_COMPLETEUPLOADRESPONSE = DESCRIPTOR.message_types_by_name['CompleteUploadResponse']
+_ABORTUPLOADREQUEST = DESCRIPTOR.message_types_by_name['AbortUploadRequest']
+_ABORTUPLOADRESPONSE = DESCRIPTOR.message_types_by_name['AbortUploadResponse']
+_MOVEUPLOADREQUEST = DESCRIPTOR.message_types_by_name['MoveUploadRequest']
+_MOVEUPLOADRESPONSE = DESCRIPTOR.message_types_by_name['MoveUploadResponse']
+_UPLOADREQUEST = DESCRIPTOR.message_types_by_name['UploadRequest']
+_UPLOADRESPONSE = DESCRIPTOR.message_types_by_name['UploadResponse']
+_GETFILEREQUEST = DESCRIPTOR.message_types_by_name['GetFileRequest']
+_GETFILERESPONSE = DESCRIPTOR.message_types_by_name['GetFileResponse']
+_DELETEFILEREQUEST = DESCRIPTOR.message_types_by_name['DeleteFileRequest']
+_DELETEFILERESPONSE = DESCRIPTOR.message_types_by_name['DeleteFileResponse']
+File = _reflection.GeneratedProtocolMessageType('File', (_message.Message,), {
+  'DESCRIPTOR' : _FILE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.File)
+  })
+_sym_db.RegisterMessage(File)
+
+MultipartUpload = _reflection.GeneratedProtocolMessageType('MultipartUpload', (_message.Message,), {
+  'DESCRIPTOR' : _MULTIPARTUPLOAD,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.MultipartUpload)
+  })
+_sym_db.RegisterMessage(MultipartUpload)
+
+Upload = _reflection.GeneratedProtocolMessageType('Upload', (_message.Message,), {
+  'DESCRIPTOR' : _UPLOAD,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.Upload)
+  })
+_sym_db.RegisterMessage(Upload)
+
+CompletedPart = _reflection.GeneratedProtocolMessageType('CompletedPart', (_message.Message,), {
+  'DESCRIPTOR' : _COMPLETEDPART,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.CompletedPart)
+  })
+_sym_db.RegisterMessage(CompletedPart)
+
+StartUploadRequest = _reflection.GeneratedProtocolMessageType('StartUploadRequest', (_message.Message,), {
+  'DESCRIPTOR' : _STARTUPLOADREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.StartUploadRequest)
+  })
+_sym_db.RegisterMessage(StartUploadRequest)
+
+StartUploadResponse = _reflection.GeneratedProtocolMessageType('StartUploadResponse', (_message.Message,), {
+  'DESCRIPTOR' : _STARTUPLOADRESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.StartUploadResponse)
+  })
+_sym_db.RegisterMessage(StartUploadResponse)
+
+CompleteUploadRequest = _reflection.GeneratedProtocolMessageType('CompleteUploadRequest', (_message.Message,), {
+  'DESCRIPTOR' : _COMPLETEUPLOADREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.CompleteUploadRequest)
+  })
+_sym_db.RegisterMessage(CompleteUploadRequest)
+
+CompleteUploadResponse = _reflection.GeneratedProtocolMessageType('CompleteUploadResponse', (_message.Message,), {
+  'DESCRIPTOR' : _COMPLETEUPLOADRESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.CompleteUploadResponse)
+  })
+_sym_db.RegisterMessage(CompleteUploadResponse)
+
+AbortUploadRequest = _reflection.GeneratedProtocolMessageType('AbortUploadRequest', (_message.Message,), {
+  'DESCRIPTOR' : _ABORTUPLOADREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.AbortUploadRequest)
+  })
+_sym_db.RegisterMessage(AbortUploadRequest)
+
+AbortUploadResponse = _reflection.GeneratedProtocolMessageType('AbortUploadResponse', (_message.Message,), {
+  'DESCRIPTOR' : _ABORTUPLOADRESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.AbortUploadResponse)
+  })
+_sym_db.RegisterMessage(AbortUploadResponse)
+
+MoveUploadRequest = _reflection.GeneratedProtocolMessageType('MoveUploadRequest', (_message.Message,), {
+  'DESCRIPTOR' : _MOVEUPLOADREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.MoveUploadRequest)
+  })
+_sym_db.RegisterMessage(MoveUploadRequest)
+
+MoveUploadResponse = _reflection.GeneratedProtocolMessageType('MoveUploadResponse', (_message.Message,), {
+  'DESCRIPTOR' : _MOVEUPLOADRESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.MoveUploadResponse)
+  })
+_sym_db.RegisterMessage(MoveUploadResponse)
+
+UploadRequest = _reflection.GeneratedProtocolMessageType('UploadRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPLOADREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.UploadRequest)
+  })
+_sym_db.RegisterMessage(UploadRequest)
+
+UploadResponse = _reflection.GeneratedProtocolMessageType('UploadResponse', (_message.Message,), {
+  'DESCRIPTOR' : _UPLOADRESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.UploadResponse)
+  })
+_sym_db.RegisterMessage(UploadResponse)
+
+GetFileRequest = _reflection.GeneratedProtocolMessageType('GetFileRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETFILEREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.GetFileRequest)
+  })
+_sym_db.RegisterMessage(GetFileRequest)
+
+GetFileResponse = _reflection.GeneratedProtocolMessageType('GetFileResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETFILERESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.GetFileResponse)
+  })
+_sym_db.RegisterMessage(GetFileResponse)
+
+DeleteFileRequest = _reflection.GeneratedProtocolMessageType('DeleteFileRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEFILEREQUEST,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.DeleteFileRequest)
+  })
+_sym_db.RegisterMessage(DeleteFileRequest)
+
+DeleteFileResponse = _reflection.GeneratedProtocolMessageType('DeleteFileResponse', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEFILERESPONSE,
+  '__module__' : 'files.files_pb2'
+  # @@protoc_insertion_point(class_scope:files.DeleteFileResponse)
+  })
+_sym_db.RegisterMessage(DeleteFileResponse)
+
+_FILES = DESCRIPTOR.services_by_name['Files']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z.git.perx.ru/perxis/perxis-go/proto/files;files'
+  _FILE._serialized_start=57
+  _FILE._serialized_end=135
+  _MULTIPARTUPLOAD._serialized_start=138
+  _MULTIPARTUPLOAD._serialized_end=276
+  _UPLOAD._serialized_start=278
+  _UPLOAD._serialized_end=333
+  _COMPLETEDPART._serialized_start=335
+  _COMPLETEDPART._serialized_end=378
+  _STARTUPLOADREQUEST._serialized_start=380
+  _STARTUPLOADREQUEST._serialized_end=440
+  _STARTUPLOADRESPONSE._serialized_start=442
+  _STARTUPLOADRESPONSE._serialized_end=503
+  _COMPLETEUPLOADREQUEST._serialized_start=505
+  _COMPLETEUPLOADREQUEST._serialized_end=568
+  _COMPLETEUPLOADRESPONSE._serialized_start=570
+  _COMPLETEUPLOADRESPONSE._serialized_end=634
+  _ABORTUPLOADREQUEST._serialized_start=636
+  _ABORTUPLOADREQUEST._serialized_end=696
+  _ABORTUPLOADRESPONSE._serialized_start=698
+  _ABORTUPLOADRESPONSE._serialized_end=719
+  _MOVEUPLOADREQUEST._serialized_start=721
+  _MOVEUPLOADREQUEST._serialized_end=780
+  _MOVEUPLOADRESPONSE._serialized_start=782
+  _MOVEUPLOADRESPONSE._serialized_end=829
+  _UPLOADREQUEST._serialized_start=831
+  _UPLOADREQUEST._serialized_end=873
+  _UPLOADRESPONSE._serialized_start=875
+  _UPLOADRESPONSE._serialized_end=922
+  _GETFILEREQUEST._serialized_start=924
+  _GETFILEREQUEST._serialized_end=952
+  _GETFILERESPONSE._serialized_start=954
+  _GETFILERESPONSE._serialized_end=998
+  _DELETEFILEREQUEST._serialized_start=1000
+  _DELETEFILEREQUEST._serialized_end=1046
+  _DELETEFILERESPONSE._serialized_start=1048
+  _DELETEFILERESPONSE._serialized_end=1068
+  _FILES._serialized_start=1071
+  _FILES._serialized_end=1551
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/files/files_pb2_grpc.py b/clients/files/files_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..35d5e59265250fafa37b4e0adeb385ca9bb3e15d
--- /dev/null
+++ b/clients/files/files_pb2_grpc.py
@@ -0,0 +1,285 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from files import files_pb2 as files_dot_files__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+class FilesStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.StartUpload = channel.unary_unary(
+                '/files.Files/StartUpload',
+                request_serializer=files_dot_files__pb2.StartUploadRequest.SerializeToString,
+                response_deserializer=files_dot_files__pb2.StartUploadResponse.FromString,
+                )
+        self.CompleteUpload = channel.unary_unary(
+                '/files.Files/CompleteUpload',
+                request_serializer=files_dot_files__pb2.CompleteUploadRequest.SerializeToString,
+                response_deserializer=files_dot_files__pb2.CompleteUploadResponse.FromString,
+                )
+        self.AbortUpload = channel.unary_unary(
+                '/files.Files/AbortUpload',
+                request_serializer=files_dot_files__pb2.AbortUploadRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.MoveUpload = channel.unary_unary(
+                '/files.Files/MoveUpload',
+                request_serializer=files_dot_files__pb2.MoveUploadRequest.SerializeToString,
+                response_deserializer=files_dot_files__pb2.MoveUploadResponse.FromString,
+                )
+        self.Upload = channel.unary_unary(
+                '/files.Files/Upload',
+                request_serializer=files_dot_files__pb2.UploadRequest.SerializeToString,
+                response_deserializer=files_dot_files__pb2.UploadResponse.FromString,
+                )
+        self.GetFile = channel.unary_unary(
+                '/files.Files/GetFile',
+                request_serializer=files_dot_files__pb2.GetFileRequest.SerializeToString,
+                response_deserializer=files_dot_files__pb2.GetFileResponse.FromString,
+                )
+        self.DeleteFile = channel.unary_unary(
+                '/files.Files/DeleteFile',
+                request_serializer=files_dot_files__pb2.DeleteFileRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class FilesServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def StartUpload(self, request, context):
+        """StartUpload - инициирует процедуру загрузки файла в файловое хранилище.
+        Используется клиентским приложением для начала загрузки файла
+        Требуемые параметры: объект `MultipartUpload` с вложенным объектом `File`, в котором
+        обязательные поля  Name и Size.
+        Возвращает объект с заполненными полями ID, PartSize, PartURLs, MimeType, Size, UploadID.
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def CompleteUpload(self, request, context):
+        """CompleteUpload - завершает процедуру загрузку файла
+        Предполагается, что в объекте, полученном из `StartUpload`, клиент должен заполнить поле
+        Parts (идентификаторами загруженных блоков (S3 ETAGs)) и передать его в `CompleteUpload`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def AbortUpload(self, request, context):
+        """AbortUpload - прерывает процедуру загрузки файла, все загруженные части файла удаляются их хранилища
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def MoveUpload(self, request, context):
+        """MoveUpload - перемещает загруженный файл из временного расположения в постоянное месторасположения.
+        После перемещение загрузки хранилище выдает новый идентификатор постоянного файла
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Upload(self, request, context):
+        """Upload - инициация загрузки файла в хранилище. Возвращает объект, содержащий подписанный URL.
+        Завершение загрузки файла осуществляется выполнением POST-запроса
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetFile(self, request, context):
+        """GetFile - получить информацию о файле (ссылку) по ID
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def DeleteFile(self, request, context):
+        """DeleteFile - удаляет файл. Если происходит удаление оригинала,
+        удаляются и все связанные структуры
+        Возможны следующие ситуации:
+        - Удаление оригинального файла. Удаляется файл и все связанные с ним файлы.
+        - Удаление связанного файла. Для удаления конкретного файла передается 'ID', 'Type' и 'Name'
+        - Удаления связанных файлов конкретного типа (например, все 'thumbnails'). Передается 'ID' и 'Type'
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_FilesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'StartUpload': grpc.unary_unary_rpc_method_handler(
+                    servicer.StartUpload,
+                    request_deserializer=files_dot_files__pb2.StartUploadRequest.FromString,
+                    response_serializer=files_dot_files__pb2.StartUploadResponse.SerializeToString,
+            ),
+            'CompleteUpload': grpc.unary_unary_rpc_method_handler(
+                    servicer.CompleteUpload,
+                    request_deserializer=files_dot_files__pb2.CompleteUploadRequest.FromString,
+                    response_serializer=files_dot_files__pb2.CompleteUploadResponse.SerializeToString,
+            ),
+            'AbortUpload': grpc.unary_unary_rpc_method_handler(
+                    servicer.AbortUpload,
+                    request_deserializer=files_dot_files__pb2.AbortUploadRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'MoveUpload': grpc.unary_unary_rpc_method_handler(
+                    servicer.MoveUpload,
+                    request_deserializer=files_dot_files__pb2.MoveUploadRequest.FromString,
+                    response_serializer=files_dot_files__pb2.MoveUploadResponse.SerializeToString,
+            ),
+            'Upload': grpc.unary_unary_rpc_method_handler(
+                    servicer.Upload,
+                    request_deserializer=files_dot_files__pb2.UploadRequest.FromString,
+                    response_serializer=files_dot_files__pb2.UploadResponse.SerializeToString,
+            ),
+            'GetFile': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetFile,
+                    request_deserializer=files_dot_files__pb2.GetFileRequest.FromString,
+                    response_serializer=files_dot_files__pb2.GetFileResponse.SerializeToString,
+            ),
+            'DeleteFile': grpc.unary_unary_rpc_method_handler(
+                    servicer.DeleteFile,
+                    request_deserializer=files_dot_files__pb2.DeleteFileRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'files.Files', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Files(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def StartUpload(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/StartUpload',
+            files_dot_files__pb2.StartUploadRequest.SerializeToString,
+            files_dot_files__pb2.StartUploadResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def CompleteUpload(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/CompleteUpload',
+            files_dot_files__pb2.CompleteUploadRequest.SerializeToString,
+            files_dot_files__pb2.CompleteUploadResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def AbortUpload(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/AbortUpload',
+            files_dot_files__pb2.AbortUploadRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def MoveUpload(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/MoveUpload',
+            files_dot_files__pb2.MoveUploadRequest.SerializeToString,
+            files_dot_files__pb2.MoveUploadResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Upload(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/Upload',
+            files_dot_files__pb2.UploadRequest.SerializeToString,
+            files_dot_files__pb2.UploadResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetFile(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/GetFile',
+            files_dot_files__pb2.GetFileRequest.SerializeToString,
+            files_dot_files__pb2.GetFileResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def DeleteFile(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/files.Files/DeleteFile',
+            files_dot_files__pb2.DeleteFileRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/images/images_pb2.py b/clients/images/images_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..ed24084b9c2c5c82a1ded7a54fb88370f2157278
--- /dev/null
+++ b/clients/images/images_pb2.py
@@ -0,0 +1,70 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: images/images.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from files import files_pb2 as files_dot_files__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x13images/images.proto\x12\x06images\x1a\x11\x66iles/files.proto\"\"\n\x05Param\x12\n\n\x02op\x18\x01 \x01(\t\x12\r\n\x05value\x18\x02 \x03(\t\"\x83\x01\n\nGetRequest\x12\x1b\n\x06source\x18\x01 \x01(\x0b\x32\x0b.files.File\x12+\n\x04opts\x18\x02 \x01(\x0b\x32\x1d.images.GetRequest.GetOptions\x1a+\n\nGetOptions\x12\x1d\n\x06params\x18\x01 \x03(\x0b\x32\r.images.Param\"*\n\x0bGetResponse\x12\x1b\n\x06result\x18\x01 \x01(\x0b\x32\x0b.files.File2:\n\x06Images\x12\x30\n\x03Get\x12\x12.images.GetRequest\x1a\x13.images.GetResponse\"\x00\x42\x32Z0git.perx.ru/perxis/perxis-go/proto/images;imagesb\x06proto3')
+
+
+
+_PARAM = DESCRIPTOR.message_types_by_name['Param']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETREQUEST_GETOPTIONS = _GETREQUEST.nested_types_by_name['GetOptions']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+Param = _reflection.GeneratedProtocolMessageType('Param', (_message.Message,), {
+  'DESCRIPTOR' : _PARAM,
+  '__module__' : 'images.images_pb2'
+  # @@protoc_insertion_point(class_scope:images.Param)
+  })
+_sym_db.RegisterMessage(Param)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+
+  'GetOptions' : _reflection.GeneratedProtocolMessageType('GetOptions', (_message.Message,), {
+    'DESCRIPTOR' : _GETREQUEST_GETOPTIONS,
+    '__module__' : 'images.images_pb2'
+    # @@protoc_insertion_point(class_scope:images.GetRequest.GetOptions)
+    })
+  ,
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'images.images_pb2'
+  # @@protoc_insertion_point(class_scope:images.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+_sym_db.RegisterMessage(GetRequest.GetOptions)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'images.images_pb2'
+  # @@protoc_insertion_point(class_scope:images.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+_IMAGES = DESCRIPTOR.services_by_name['Images']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z0git.perx.ru/perxis/perxis-go/proto/images;images'
+  _PARAM._serialized_start=50
+  _PARAM._serialized_end=84
+  _GETREQUEST._serialized_start=87
+  _GETREQUEST._serialized_end=218
+  _GETREQUEST_GETOPTIONS._serialized_start=175
+  _GETREQUEST_GETOPTIONS._serialized_end=218
+  _GETRESPONSE._serialized_start=220
+  _GETRESPONSE._serialized_end=262
+  _IMAGES._serialized_start=264
+  _IMAGES._serialized_end=322
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/images/images_pb2_grpc.py b/clients/images/images_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..01e83b7d0039744b6e048a082aa7c8f66b6e0c74
--- /dev/null
+++ b/clients/images/images_pb2_grpc.py
@@ -0,0 +1,77 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from images import images_pb2 as images_dot_images__pb2
+
+
+class ImagesStub(object):
+    """ImageService - сервис для обработки изображений
+    """
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Get = channel.unary_unary(
+                '/images.Images/Get',
+                request_serializer=images_dot_images__pb2.GetRequest.SerializeToString,
+                response_deserializer=images_dot_images__pb2.GetResponse.FromString,
+                )
+
+
+class ImagesServicer(object):
+    """ImageService - сервис для обработки изображений
+    """
+
+    def Get(self, request, context):
+        """Get - к файлу, идентификатор которого передан в запросе, применяются параметры.
+        Может быть передано несколько параметров, порядок учитывается при обработке
+        Допустимые операции:
+        - crop (ширина, высота)
+        - fit (ширина, высота)
+        - resize (ширина, высота)
+        - format (один из форматов [ png, jpeg, jpg, gif ]
+        Возвращает объект File с заполненным полем File.URL
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ImagesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=images_dot_images__pb2.GetRequest.FromString,
+                    response_serializer=images_dot_images__pb2.GetResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'images.Images', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Images(object):
+    """ImageService - сервис для обработки изображений
+    """
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/images.Images/Get',
+            images_dot_images__pb2.GetRequest.SerializeToString,
+            images_dot_images__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/invitations/invitations_pb2.py b/clients/invitations/invitations_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..ae7411e5dfea0a905480ed74133d0b76a2f564ff
--- /dev/null
+++ b/clients/invitations/invitations_pb2.py
@@ -0,0 +1,181 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: invitations/invitations.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import timestamp_pb2 as google_dot_protobuf_dot_timestamp__pb2
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from google.protobuf import wrappers_pb2 as google_dot_protobuf_dot_wrappers__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1dinvitations/invitations.proto\x12\x13\x63ontent.invitations\x1a\x1fgoogle/protobuf/timestamp.proto\x1a\x1bgoogle/protobuf/empty.proto\x1a\x1egoogle/protobuf/wrappers.proto\"\xca\x01\n\nInvitation\x12\n\n\x02id\x18\x01 \x01(\t\x12\r\n\x05\x65mail\x18\x02 \x01(\t\x12\x0e\n\x06org_id\x18\x03 \x01(\t\x12\x10\n\x08space_id\x18\x04 \x01(\t\x12\x10\n\x08owner_id\x18\x05 \x01(\t\x12\x0c\n\x04role\x18\x06 \x01(\t\x12.\n\ncreated_at\x18\x07 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12/\n\x0bvalid_until\x18\x08 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\"e\n\x06\x46ilter\x12\n\n\x02id\x18\x01 \x03(\t\x12\r\n\x05\x65mail\x18\x02 \x03(\t\x12\x0e\n\x06org_id\x18\x03 \x03(\t\x12\x10\n\x08space_id\x18\x04 \x03(\t\x12\x10\n\x08owner_id\x18\x05 \x03(\t\x12\x0c\n\x04role\x18\x06 \x03(\t\"@\n\x0b\x46indOptions\x12\x0c\n\x04sort\x18\x01 \x03(\t\x12\x10\n\x08page_num\x18\x02 \x01(\x05\x12\x11\n\tpage_size\x18\x03 \x01(\x05\"D\n\rCreateRequest\x12\x33\n\ninvitation\x18\x01 \x01(\x0b\x32\x1f.content.invitations.Invitation\"E\n\x0e\x43reateResponse\x12\x33\n\ninvitation\x18\x01 \x01(\x0b\x32\x1f.content.invitations.Invitation\"#\n\nGetRequest\x12\x15\n\rinvitation_id\x18\x01 \x01(\t\"B\n\x0bGetResponse\x12\x33\n\ninvitation\x18\x01 \x01(\x0b\x32\x1f.content.invitations.Invitation\"\xdb\x01\n\rUpdateRequest\x12\x15\n\rinvitation_id\x18\x01 \x01(\t\x12(\n\x04sent\x18\x02 \x01(\x0b\x32\x1a.google.protobuf.BoolValue\x12+\n\x07pending\x18\x03 \x01(\x0b\x32\x1a.google.protobuf.BoolValue\x12+\n\x07sent_at\x18\x04 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12/\n\x0bvalid_until\x18\x05 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\"\x1a\n\x18UpdateInvitationResponse\"7\n\rAcceptRequest\x12\x15\n\rinvitation_id\x18\x01 \x01(\t\x12\x0f\n\x07user_id\x18\x02 \x01(\t\"\x1a\n\x18\x41\x63\x63\x65ptInvitationResponse\"j\n\x0b\x46indRequest\x12+\n\x06\x66ilter\x18\x01 \x01(\x0b\x32\x1b.content.invitations.Filter\x12.\n\x04opts\x18\x02 \x01(\x0b\x32 .content.invitations.FindOptions\"S\n\x0c\x46indResponse\x12\x34\n\x0binvitations\x18\x01 \x03(\x0b\x32\x1f.content.invitations.Invitation\x12\r\n\x05total\x18\x02 \x01(\x03\"&\n\rDeleteRequest\x12\x15\n\rinvitation_id\x18\x01 \x01(\t\"\x1f\n\x1d\x44\x65leteSpaceInvitationResponse2\x8d\x03\n\x0bInvitations\x12S\n\x06\x43reate\x12\".content.invitations.CreateRequest\x1a#.content.invitations.CreateResponse\"\x00\x12J\n\x03Get\x12\x1f.content.invitations.GetRequest\x1a .content.invitations.GetResponse\"\x00\x12\x46\n\x06\x41\x63\x63\x65pt\x12\".content.invitations.AcceptRequest\x1a\x16.google.protobuf.Empty\"\x00\x12M\n\x04\x46ind\x12 .content.invitations.FindRequest\x1a!.content.invitations.FindResponse\"\x00\x12\x46\n\x06\x44\x65lete\x12\".content.invitations.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42<Z:git.perx.ru/perxis/perxis-go/proto/invitations;invitationsb\x06proto3')
+
+
+
+_INVITATION = DESCRIPTOR.message_types_by_name['Invitation']
+_FILTER = DESCRIPTOR.message_types_by_name['Filter']
+_FINDOPTIONS = DESCRIPTOR.message_types_by_name['FindOptions']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_UPDATEINVITATIONRESPONSE = DESCRIPTOR.message_types_by_name['UpdateInvitationResponse']
+_ACCEPTREQUEST = DESCRIPTOR.message_types_by_name['AcceptRequest']
+_ACCEPTINVITATIONRESPONSE = DESCRIPTOR.message_types_by_name['AcceptInvitationResponse']
+_FINDREQUEST = DESCRIPTOR.message_types_by_name['FindRequest']
+_FINDRESPONSE = DESCRIPTOR.message_types_by_name['FindResponse']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_DELETESPACEINVITATIONRESPONSE = DESCRIPTOR.message_types_by_name['DeleteSpaceInvitationResponse']
+Invitation = _reflection.GeneratedProtocolMessageType('Invitation', (_message.Message,), {
+  'DESCRIPTOR' : _INVITATION,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.Invitation)
+  })
+_sym_db.RegisterMessage(Invitation)
+
+Filter = _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+  'DESCRIPTOR' : _FILTER,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.Filter)
+  })
+_sym_db.RegisterMessage(Filter)
+
+FindOptions = _reflection.GeneratedProtocolMessageType('FindOptions', (_message.Message,), {
+  'DESCRIPTOR' : _FINDOPTIONS,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.FindOptions)
+  })
+_sym_db.RegisterMessage(FindOptions)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+UpdateInvitationResponse = _reflection.GeneratedProtocolMessageType('UpdateInvitationResponse', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEINVITATIONRESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.UpdateInvitationResponse)
+  })
+_sym_db.RegisterMessage(UpdateInvitationResponse)
+
+AcceptRequest = _reflection.GeneratedProtocolMessageType('AcceptRequest', (_message.Message,), {
+  'DESCRIPTOR' : _ACCEPTREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.AcceptRequest)
+  })
+_sym_db.RegisterMessage(AcceptRequest)
+
+AcceptInvitationResponse = _reflection.GeneratedProtocolMessageType('AcceptInvitationResponse', (_message.Message,), {
+  'DESCRIPTOR' : _ACCEPTINVITATIONRESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.AcceptInvitationResponse)
+  })
+_sym_db.RegisterMessage(AcceptInvitationResponse)
+
+FindRequest = _reflection.GeneratedProtocolMessageType('FindRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.FindRequest)
+  })
+_sym_db.RegisterMessage(FindRequest)
+
+FindResponse = _reflection.GeneratedProtocolMessageType('FindResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDRESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.FindResponse)
+  })
+_sym_db.RegisterMessage(FindResponse)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+DeleteSpaceInvitationResponse = _reflection.GeneratedProtocolMessageType('DeleteSpaceInvitationResponse', (_message.Message,), {
+  'DESCRIPTOR' : _DELETESPACEINVITATIONRESPONSE,
+  '__module__' : 'invitations.invitations_pb2'
+  # @@protoc_insertion_point(class_scope:content.invitations.DeleteSpaceInvitationResponse)
+  })
+_sym_db.RegisterMessage(DeleteSpaceInvitationResponse)
+
+_INVITATIONS = DESCRIPTOR.services_by_name['Invitations']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z:git.perx.ru/perxis/perxis-go/proto/invitations;invitations'
+  _INVITATION._serialized_start=149
+  _INVITATION._serialized_end=351
+  _FILTER._serialized_start=353
+  _FILTER._serialized_end=454
+  _FINDOPTIONS._serialized_start=456
+  _FINDOPTIONS._serialized_end=520
+  _CREATEREQUEST._serialized_start=522
+  _CREATEREQUEST._serialized_end=590
+  _CREATERESPONSE._serialized_start=592
+  _CREATERESPONSE._serialized_end=661
+  _GETREQUEST._serialized_start=663
+  _GETREQUEST._serialized_end=698
+  _GETRESPONSE._serialized_start=700
+  _GETRESPONSE._serialized_end=766
+  _UPDATEREQUEST._serialized_start=769
+  _UPDATEREQUEST._serialized_end=988
+  _UPDATEINVITATIONRESPONSE._serialized_start=990
+  _UPDATEINVITATIONRESPONSE._serialized_end=1016
+  _ACCEPTREQUEST._serialized_start=1018
+  _ACCEPTREQUEST._serialized_end=1073
+  _ACCEPTINVITATIONRESPONSE._serialized_start=1075
+  _ACCEPTINVITATIONRESPONSE._serialized_end=1101
+  _FINDREQUEST._serialized_start=1103
+  _FINDREQUEST._serialized_end=1209
+  _FINDRESPONSE._serialized_start=1211
+  _FINDRESPONSE._serialized_end=1294
+  _DELETEREQUEST._serialized_start=1296
+  _DELETEREQUEST._serialized_end=1334
+  _DELETESPACEINVITATIONRESPONSE._serialized_start=1336
+  _DELETESPACEINVITATIONRESPONSE._serialized_end=1367
+  _INVITATIONS._serialized_start=1370
+  _INVITATIONS._serialized_end=1767
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/invitations/invitations_pb2_grpc.py b/clients/invitations/invitations_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..a0a77603972df2804fd754f8b1abdd32b9332231
--- /dev/null
+++ b/clients/invitations/invitations_pb2_grpc.py
@@ -0,0 +1,206 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from invitations import invitations_pb2 as invitations_dot_invitations__pb2
+
+
+class InvitationsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.invitations.Invitations/Create',
+                request_serializer=invitations_dot_invitations__pb2.CreateRequest.SerializeToString,
+                response_deserializer=invitations_dot_invitations__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.invitations.Invitations/Get',
+                request_serializer=invitations_dot_invitations__pb2.GetRequest.SerializeToString,
+                response_deserializer=invitations_dot_invitations__pb2.GetResponse.FromString,
+                )
+        self.Accept = channel.unary_unary(
+                '/content.invitations.Invitations/Accept',
+                request_serializer=invitations_dot_invitations__pb2.AcceptRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Find = channel.unary_unary(
+                '/content.invitations.Invitations/Find',
+                request_serializer=invitations_dot_invitations__pb2.FindRequest.SerializeToString,
+                response_deserializer=invitations_dot_invitations__pb2.FindResponse.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.invitations.Invitations/Delete',
+                request_serializer=invitations_dot_invitations__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class InvitationsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Получить данные о приглашении
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Accept(self, request, context):
+        """Обновить данные о приглашении
+        rpc Update(UpdateRequest) returns(google.protobuf.Empty) {}
+
+        Принять приглашение
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Find(self, request, context):
+        """Получить список приглашений для пространства
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Удалить приглашение
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_InvitationsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=invitations_dot_invitations__pb2.CreateRequest.FromString,
+                    response_serializer=invitations_dot_invitations__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=invitations_dot_invitations__pb2.GetRequest.FromString,
+                    response_serializer=invitations_dot_invitations__pb2.GetResponse.SerializeToString,
+            ),
+            'Accept': grpc.unary_unary_rpc_method_handler(
+                    servicer.Accept,
+                    request_deserializer=invitations_dot_invitations__pb2.AcceptRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Find': grpc.unary_unary_rpc_method_handler(
+                    servicer.Find,
+                    request_deserializer=invitations_dot_invitations__pb2.FindRequest.FromString,
+                    response_serializer=invitations_dot_invitations__pb2.FindResponse.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=invitations_dot_invitations__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.invitations.Invitations', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Invitations(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.invitations.Invitations/Create',
+            invitations_dot_invitations__pb2.CreateRequest.SerializeToString,
+            invitations_dot_invitations__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.invitations.Invitations/Get',
+            invitations_dot_invitations__pb2.GetRequest.SerializeToString,
+            invitations_dot_invitations__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Accept(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.invitations.Invitations/Accept',
+            invitations_dot_invitations__pb2.AcceptRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Find(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.invitations.Invitations/Find',
+            invitations_dot_invitations__pb2.FindRequest.SerializeToString,
+            invitations_dot_invitations__pb2.FindResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.invitations.Invitations/Delete',
+            invitations_dot_invitations__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/items/items_pb2.py b/clients/items/items_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..0078dd557335993642037c09305c9eeeb78e12d4
--- /dev/null
+++ b/clients/items/items_pb2.py
@@ -0,0 +1,458 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: items/items.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+from google.protobuf import timestamp_pb2 as google_dot_protobuf_dot_timestamp__pb2
+from google.protobuf import struct_pb2 as google_dot_protobuf_dot_struct__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x11items/items.proto\x12\rcontent.items\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\x1a\x1fgoogle/protobuf/timestamp.proto\x1a\x1cgoogle/protobuf/struct.proto\"\'\n\x05\x45rror\x12\x0f\n\x07message\x18\x01 \x01(\t\x12\r\n\x05\x66ield\x18\x02 \x01(\t\"3\n\x0b\x44\x65\x63odeError\x12$\n\x06\x65rrors\x18\x02 \x03(\x0b\x32\x14.content.items.Error\"7\n\x0fValidationError\x12$\n\x06\x65rrors\x18\x02 \x03(\x0b\x32\x14.content.items.Error\"9\n\x11ModificationError\x12$\n\x06\x65rrors\x18\x02 \x03(\x0b\x32\x14.content.items.Error\"g\n\x0bPermissions\x12\x0c\n\x04\x65\x64it\x18\x01 \x01(\x08\x12\x0f\n\x07\x61rchive\x18\x02 \x01(\x08\x12\x0f\n\x07publish\x18\x03 \x01(\x08\x12\x13\n\x0bsoft_delete\x18\x04 \x01(\x08\x12\x13\n\x0bhard_delete\x18\x05 \x01(\x08\"\xb8\x06\n\x04Item\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x03 \x01(\t\x12\x15\n\rcollection_id\x18\x04 \x01(\t\x12(\n\x05state\x18\x05 \x01(\x0e\x32\x19.content.items.Item.State\x12\x32\n\x0e\x63reated_rev_at\x18\x06 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12\x12\n\ncreated_by\x18\x07 \x01(\t\x12.\n\ncreated_at\x18\x08 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12\x12\n\nupdated_by\x18\t \x01(\t\x12.\n\nupdated_at\x18\n \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12%\n\x04\x64\x61ta\x18\x0b \x01(\x0b\x32\x17.google.protobuf.Struct\x12;\n\x0ctranslations\x18\x0c \x03(\x0b\x32%.content.items.Item.TranslationsEntry\x12\x13\n\x0brevision_id\x18\r \x01(\t\x12\x30\n\x0cpublished_at\x18\x0e \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12\x14\n\x0cpublished_by\x18\x0f \x01(\t\x12/\n\x0b\x61rchived_at\x18\x10 \x01(\x0b\x32\x1a.google.protobuf.Timestamp\x12\x13\n\x0b\x61rchived_by\x18\x11 \x01(\t\x12\x0e\n\x06locale\x18\x12 \x01(\t\x12\x0f\n\x07\x64\x65leted\x18\x13 \x01(\x08\x12\x0e\n\x06hidden\x18\x14 \x01(\x08\x12\x10\n\x08template\x18\x15 \x01(\x08\x12/\n\x0bpermissions\x18\x16 \x01(\x0b\x32\x1a.content.items.Permissions\x1aL\n\x11TranslationsEntry\x12\x0b\n\x03key\x18\x01 \x01(\t\x12&\n\x05value\x18\x02 \x01(\x0b\x32\x17.google.protobuf.Struct:\x02\x38\x01\"<\n\x05State\x12\t\n\x05\x44RAFT\x10\x00\x12\r\n\tPUBLISHED\x10\x01\x12\x0b\n\x07\x43HANGED\x10\x02\x12\x0c\n\x08\x41RCHIVED\x10\x03\"=\n\x06\x46ilter\x12\n\n\x02id\x18\x01 \x03(\t\x12\x1c\n\x04\x64\x61ta\x18\x02 \x03(\x0b\x32\x0e.common.Filter\x12\t\n\x01q\x18\x03 \x03(\t\"%\n\rCreateOptions\x12\x14\n\x0cupdate_attrs\x18\x01 \x01(\x08\"x\n\x0b\x46indOptions\x12$\n\x07options\x18\x02 \x01(\x0b\x32\x13.common.FindOptions\x12\x0f\n\x07\x64\x65leted\x18\x03 \x01(\x08\x12\x0f\n\x07regular\x18\x04 \x01(\x08\x12\x0e\n\x06hidden\x18\x05 \x01(\x08\x12\x11\n\ttemplates\x18\x06 \x01(\x08\"%\n\rUpdateOptions\x12\x14\n\x0cupdate_attrs\x18\x01 \x01(\x08\"(\n\x13GetPublishedOptions\x12\x11\n\tlocale_id\x18\x01 \x01(\t\"\x1e\n\rDeleteOptions\x12\r\n\x05\x65rase\x18\x01 \x01(\x08\"&\n\x0ePublishOptions\x12\x14\n\x0cupdate_attrs\x18\x01 \x01(\x08\"\x83\x01\n\x14\x46indPublishedOptions\x12$\n\x07options\x18\x02 \x01(\x0b\x32\x13.common.FindOptions\x12\x11\n\tlocale_id\x18\x03 \x01(\t\x12\x0f\n\x07regular\x18\x04 \x01(\x08\x12\x0e\n\x06hidden\x18\x05 \x01(\x08\x12\x11\n\ttemplates\x18\x06 \x01(\x08\";\n\x13\x46indArchivedOptions\x12$\n\x07options\x18\x02 \x01(\x0b\x32\x13.common.FindOptions\"<\n\x14ListRevisionsOptions\x12$\n\x07options\x18\x02 \x01(\x0b\x32\x13.common.FindOptions\"a\n\rCreateRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\x12-\n\x07options\x18\x02 \x01(\x0b\x32\x1c.content.items.CreateOptions\"6\n\x0e\x43reateResponse\x12$\n\x07\x63reated\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"6\n\x11IntrospectRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\x8b\x01\n\x12IntrospectResponse\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\x12\x0e\n\x06schema\x18\x02 \x01(\t\x12\x42\n\x11validation_errors\x18\x03 \x03(\x0b\x32\'.common.Error.BadRequest.FieldViolation\"V\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x04 \x01(\t\"0\n\x0bGetResponse\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\x9a\x01\n\x0b\x46indRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12%\n\x06\x66ilter\x18\x04 \x01(\x0b\x32\x15.content.items.Filter\x12+\n\x07options\x18\x05 \x01(\x0b\x32\x1a.content.items.FindOptions\"A\n\x0c\x46indResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\x12\r\n\x05total\x18\x02 \x01(\x05\"a\n\rUpdateRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\x12-\n\x07options\x18\x02 \x01(\x0b\x32\x1c.content.items.UpdateOptions\"\x88\x01\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x04 \x01(\t\x12-\n\x07options\x18\x05 \x01(\x0b\x32\x1c.content.items.DeleteOptions\"[\n\x0fUndeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x04 \x01(\t\"c\n\x0ePublishRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\x12.\n\x07options\x18\x02 \x01(\x0b\x32\x1d.content.items.PublishOptions\"5\n\x10UnpublishRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\x94\x01\n\x13GetPublishedRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x05 \x01(\t\x12\x33\n\x07options\x18\n \x01(\x0b\x32\".content.items.GetPublishedOptions\"9\n\x14GetPublishedResponse\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\xac\x01\n\x14\x46indPublishedRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12%\n\x06\x66ilter\x18\x04 \x01(\x0b\x32\x15.content.items.Filter\x12\x34\n\x07options\x18\n \x01(\x0b\x32#.content.items.FindPublishedOptions\"J\n\x15\x46indPublishedResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\x12\r\n\x05total\x18\x02 \x01(\x05\"s\n\x12GetRevisionRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x04 \x01(\t\x12\x13\n\x0brevision_id\x18\x05 \x01(\t\"8\n\x13GetRevisionResponse\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\x96\x01\n\x14ListRevisionsRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12\x0f\n\x07item_id\x18\x04 \x01(\t\x12\x34\n\x07options\x18\n \x01(\x0b\x32#.content.items.ListRevisionsOptions\";\n\x15ListRevisionsResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\"3\n\x0e\x41rchiveRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"5\n\x10UnarchiveRequest\x12!\n\x04item\x18\x01 \x01(\x0b\x32\x13.content.items.Item\"\xaa\x01\n\x13\x46indArchivedRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x15\n\rcollection_id\x18\x03 \x01(\t\x12%\n\x06\x66ilter\x18\x04 \x01(\x0b\x32\x15.content.items.Filter\x12\x33\n\x07options\x18\x06 \x01(\x0b\x32\".content.items.FindArchivedOptions\"I\n\x14\x46indArchivedResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\x12\r\n\x05total\x18\x02 \x01(\x05\x32\xd4\t\n\x05Items\x12G\n\x06\x43reate\x12\x1c.content.items.CreateRequest\x1a\x1d.content.items.CreateResponse\"\x00\x12S\n\nIntrospect\x12 .content.items.IntrospectRequest\x1a!.content.items.IntrospectResponse\"\x00\x12>\n\x03Get\x12\x19.content.items.GetRequest\x1a\x1a.content.items.GetResponse\"\x00\x12\x41\n\x04\x46ind\x12\x1a.content.items.FindRequest\x1a\x1b.content.items.FindResponse\"\x00\x12@\n\x06Update\x12\x1c.content.items.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12@\n\x06\x44\x65lete\x12\x1c.content.items.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x44\n\x08Undelete\x12\x1e.content.items.UndeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x42\n\x07Publish\x12\x1d.content.items.PublishRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x46\n\tUnpublish\x12\x1f.content.items.UnpublishRequest\x1a\x16.google.protobuf.Empty\"\x00\x12Y\n\x0cGetPublished\x12\".content.items.GetPublishedRequest\x1a#.content.items.GetPublishedResponse\"\x00\x12\\\n\rFindPublished\x12#.content.items.FindPublishedRequest\x1a$.content.items.FindPublishedResponse\"\x00\x12V\n\x0bGetRevision\x12!.content.items.GetRevisionRequest\x1a\".content.items.GetRevisionResponse\"\x00\x12\\\n\rListRevisions\x12#.content.items.ListRevisionsRequest\x1a$.content.items.ListRevisionsResponse\"\x00\x12\x42\n\x07\x41rchive\x12\x1d.content.items.ArchiveRequest\x1a\x16.google.protobuf.Empty\"\x00\x12Y\n\x0c\x46indArchived\x12\".content.items.FindArchivedRequest\x1a#.content.items.FindArchivedResponse\"\x00\x12\x46\n\tUnarchive\x12\x1f.content.items.UnarchiveRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x30Z.git.perx.ru/perxis/perxis-go/proto/items;itemsb\x06proto3')
+
+
+
+_ERROR = DESCRIPTOR.message_types_by_name['Error']
+_DECODEERROR = DESCRIPTOR.message_types_by_name['DecodeError']
+_VALIDATIONERROR = DESCRIPTOR.message_types_by_name['ValidationError']
+_MODIFICATIONERROR = DESCRIPTOR.message_types_by_name['ModificationError']
+_PERMISSIONS = DESCRIPTOR.message_types_by_name['Permissions']
+_ITEM = DESCRIPTOR.message_types_by_name['Item']
+_ITEM_TRANSLATIONSENTRY = _ITEM.nested_types_by_name['TranslationsEntry']
+_FILTER = DESCRIPTOR.message_types_by_name['Filter']
+_CREATEOPTIONS = DESCRIPTOR.message_types_by_name['CreateOptions']
+_FINDOPTIONS = DESCRIPTOR.message_types_by_name['FindOptions']
+_UPDATEOPTIONS = DESCRIPTOR.message_types_by_name['UpdateOptions']
+_GETPUBLISHEDOPTIONS = DESCRIPTOR.message_types_by_name['GetPublishedOptions']
+_DELETEOPTIONS = DESCRIPTOR.message_types_by_name['DeleteOptions']
+_PUBLISHOPTIONS = DESCRIPTOR.message_types_by_name['PublishOptions']
+_FINDPUBLISHEDOPTIONS = DESCRIPTOR.message_types_by_name['FindPublishedOptions']
+_FINDARCHIVEDOPTIONS = DESCRIPTOR.message_types_by_name['FindArchivedOptions']
+_LISTREVISIONSOPTIONS = DESCRIPTOR.message_types_by_name['ListRevisionsOptions']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_INTROSPECTREQUEST = DESCRIPTOR.message_types_by_name['IntrospectRequest']
+_INTROSPECTRESPONSE = DESCRIPTOR.message_types_by_name['IntrospectResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_FINDREQUEST = DESCRIPTOR.message_types_by_name['FindRequest']
+_FINDRESPONSE = DESCRIPTOR.message_types_by_name['FindResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_UNDELETEREQUEST = DESCRIPTOR.message_types_by_name['UndeleteRequest']
+_PUBLISHREQUEST = DESCRIPTOR.message_types_by_name['PublishRequest']
+_UNPUBLISHREQUEST = DESCRIPTOR.message_types_by_name['UnpublishRequest']
+_GETPUBLISHEDREQUEST = DESCRIPTOR.message_types_by_name['GetPublishedRequest']
+_GETPUBLISHEDRESPONSE = DESCRIPTOR.message_types_by_name['GetPublishedResponse']
+_FINDPUBLISHEDREQUEST = DESCRIPTOR.message_types_by_name['FindPublishedRequest']
+_FINDPUBLISHEDRESPONSE = DESCRIPTOR.message_types_by_name['FindPublishedResponse']
+_GETREVISIONREQUEST = DESCRIPTOR.message_types_by_name['GetRevisionRequest']
+_GETREVISIONRESPONSE = DESCRIPTOR.message_types_by_name['GetRevisionResponse']
+_LISTREVISIONSREQUEST = DESCRIPTOR.message_types_by_name['ListRevisionsRequest']
+_LISTREVISIONSRESPONSE = DESCRIPTOR.message_types_by_name['ListRevisionsResponse']
+_ARCHIVEREQUEST = DESCRIPTOR.message_types_by_name['ArchiveRequest']
+_UNARCHIVEREQUEST = DESCRIPTOR.message_types_by_name['UnarchiveRequest']
+_FINDARCHIVEDREQUEST = DESCRIPTOR.message_types_by_name['FindArchivedRequest']
+_FINDARCHIVEDRESPONSE = DESCRIPTOR.message_types_by_name['FindArchivedResponse']
+_ITEM_STATE = _ITEM.enum_types_by_name['State']
+Error = _reflection.GeneratedProtocolMessageType('Error', (_message.Message,), {
+  'DESCRIPTOR' : _ERROR,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.Error)
+  })
+_sym_db.RegisterMessage(Error)
+
+DecodeError = _reflection.GeneratedProtocolMessageType('DecodeError', (_message.Message,), {
+  'DESCRIPTOR' : _DECODEERROR,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.DecodeError)
+  })
+_sym_db.RegisterMessage(DecodeError)
+
+ValidationError = _reflection.GeneratedProtocolMessageType('ValidationError', (_message.Message,), {
+  'DESCRIPTOR' : _VALIDATIONERROR,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ValidationError)
+  })
+_sym_db.RegisterMessage(ValidationError)
+
+ModificationError = _reflection.GeneratedProtocolMessageType('ModificationError', (_message.Message,), {
+  'DESCRIPTOR' : _MODIFICATIONERROR,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ModificationError)
+  })
+_sym_db.RegisterMessage(ModificationError)
+
+Permissions = _reflection.GeneratedProtocolMessageType('Permissions', (_message.Message,), {
+  'DESCRIPTOR' : _PERMISSIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.Permissions)
+  })
+_sym_db.RegisterMessage(Permissions)
+
+Item = _reflection.GeneratedProtocolMessageType('Item', (_message.Message,), {
+
+  'TranslationsEntry' : _reflection.GeneratedProtocolMessageType('TranslationsEntry', (_message.Message,), {
+    'DESCRIPTOR' : _ITEM_TRANSLATIONSENTRY,
+    '__module__' : 'items.items_pb2'
+    # @@protoc_insertion_point(class_scope:content.items.Item.TranslationsEntry)
+    })
+  ,
+  'DESCRIPTOR' : _ITEM,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.Item)
+  })
+_sym_db.RegisterMessage(Item)
+_sym_db.RegisterMessage(Item.TranslationsEntry)
+
+Filter = _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+  'DESCRIPTOR' : _FILTER,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.Filter)
+  })
+_sym_db.RegisterMessage(Filter)
+
+CreateOptions = _reflection.GeneratedProtocolMessageType('CreateOptions', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.CreateOptions)
+  })
+_sym_db.RegisterMessage(CreateOptions)
+
+FindOptions = _reflection.GeneratedProtocolMessageType('FindOptions', (_message.Message,), {
+  'DESCRIPTOR' : _FINDOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindOptions)
+  })
+_sym_db.RegisterMessage(FindOptions)
+
+UpdateOptions = _reflection.GeneratedProtocolMessageType('UpdateOptions', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.UpdateOptions)
+  })
+_sym_db.RegisterMessage(UpdateOptions)
+
+GetPublishedOptions = _reflection.GeneratedProtocolMessageType('GetPublishedOptions', (_message.Message,), {
+  'DESCRIPTOR' : _GETPUBLISHEDOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetPublishedOptions)
+  })
+_sym_db.RegisterMessage(GetPublishedOptions)
+
+DeleteOptions = _reflection.GeneratedProtocolMessageType('DeleteOptions', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.DeleteOptions)
+  })
+_sym_db.RegisterMessage(DeleteOptions)
+
+PublishOptions = _reflection.GeneratedProtocolMessageType('PublishOptions', (_message.Message,), {
+  'DESCRIPTOR' : _PUBLISHOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.PublishOptions)
+  })
+_sym_db.RegisterMessage(PublishOptions)
+
+FindPublishedOptions = _reflection.GeneratedProtocolMessageType('FindPublishedOptions', (_message.Message,), {
+  'DESCRIPTOR' : _FINDPUBLISHEDOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindPublishedOptions)
+  })
+_sym_db.RegisterMessage(FindPublishedOptions)
+
+FindArchivedOptions = _reflection.GeneratedProtocolMessageType('FindArchivedOptions', (_message.Message,), {
+  'DESCRIPTOR' : _FINDARCHIVEDOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindArchivedOptions)
+  })
+_sym_db.RegisterMessage(FindArchivedOptions)
+
+ListRevisionsOptions = _reflection.GeneratedProtocolMessageType('ListRevisionsOptions', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREVISIONSOPTIONS,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ListRevisionsOptions)
+  })
+_sym_db.RegisterMessage(ListRevisionsOptions)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+IntrospectRequest = _reflection.GeneratedProtocolMessageType('IntrospectRequest', (_message.Message,), {
+  'DESCRIPTOR' : _INTROSPECTREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.IntrospectRequest)
+  })
+_sym_db.RegisterMessage(IntrospectRequest)
+
+IntrospectResponse = _reflection.GeneratedProtocolMessageType('IntrospectResponse', (_message.Message,), {
+  'DESCRIPTOR' : _INTROSPECTRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.IntrospectResponse)
+  })
+_sym_db.RegisterMessage(IntrospectResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+FindRequest = _reflection.GeneratedProtocolMessageType('FindRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindRequest)
+  })
+_sym_db.RegisterMessage(FindRequest)
+
+FindResponse = _reflection.GeneratedProtocolMessageType('FindResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindResponse)
+  })
+_sym_db.RegisterMessage(FindResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+UndeleteRequest = _reflection.GeneratedProtocolMessageType('UndeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UNDELETEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.UndeleteRequest)
+  })
+_sym_db.RegisterMessage(UndeleteRequest)
+
+PublishRequest = _reflection.GeneratedProtocolMessageType('PublishRequest', (_message.Message,), {
+  'DESCRIPTOR' : _PUBLISHREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.PublishRequest)
+  })
+_sym_db.RegisterMessage(PublishRequest)
+
+UnpublishRequest = _reflection.GeneratedProtocolMessageType('UnpublishRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UNPUBLISHREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.UnpublishRequest)
+  })
+_sym_db.RegisterMessage(UnpublishRequest)
+
+GetPublishedRequest = _reflection.GeneratedProtocolMessageType('GetPublishedRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETPUBLISHEDREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetPublishedRequest)
+  })
+_sym_db.RegisterMessage(GetPublishedRequest)
+
+GetPublishedResponse = _reflection.GeneratedProtocolMessageType('GetPublishedResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETPUBLISHEDRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetPublishedResponse)
+  })
+_sym_db.RegisterMessage(GetPublishedResponse)
+
+FindPublishedRequest = _reflection.GeneratedProtocolMessageType('FindPublishedRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDPUBLISHEDREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindPublishedRequest)
+  })
+_sym_db.RegisterMessage(FindPublishedRequest)
+
+FindPublishedResponse = _reflection.GeneratedProtocolMessageType('FindPublishedResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDPUBLISHEDRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindPublishedResponse)
+  })
+_sym_db.RegisterMessage(FindPublishedResponse)
+
+GetRevisionRequest = _reflection.GeneratedProtocolMessageType('GetRevisionRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREVISIONREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetRevisionRequest)
+  })
+_sym_db.RegisterMessage(GetRevisionRequest)
+
+GetRevisionResponse = _reflection.GeneratedProtocolMessageType('GetRevisionResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETREVISIONRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.GetRevisionResponse)
+  })
+_sym_db.RegisterMessage(GetRevisionResponse)
+
+ListRevisionsRequest = _reflection.GeneratedProtocolMessageType('ListRevisionsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREVISIONSREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ListRevisionsRequest)
+  })
+_sym_db.RegisterMessage(ListRevisionsRequest)
+
+ListRevisionsResponse = _reflection.GeneratedProtocolMessageType('ListRevisionsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREVISIONSRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ListRevisionsResponse)
+  })
+_sym_db.RegisterMessage(ListRevisionsResponse)
+
+ArchiveRequest = _reflection.GeneratedProtocolMessageType('ArchiveRequest', (_message.Message,), {
+  'DESCRIPTOR' : _ARCHIVEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.ArchiveRequest)
+  })
+_sym_db.RegisterMessage(ArchiveRequest)
+
+UnarchiveRequest = _reflection.GeneratedProtocolMessageType('UnarchiveRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UNARCHIVEREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.UnarchiveRequest)
+  })
+_sym_db.RegisterMessage(UnarchiveRequest)
+
+FindArchivedRequest = _reflection.GeneratedProtocolMessageType('FindArchivedRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDARCHIVEDREQUEST,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindArchivedRequest)
+  })
+_sym_db.RegisterMessage(FindArchivedRequest)
+
+FindArchivedResponse = _reflection.GeneratedProtocolMessageType('FindArchivedResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDARCHIVEDRESPONSE,
+  '__module__' : 'items.items_pb2'
+  # @@protoc_insertion_point(class_scope:content.items.FindArchivedResponse)
+  })
+_sym_db.RegisterMessage(FindArchivedResponse)
+
+_ITEMS = DESCRIPTOR.services_by_name['Items']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z.git.perx.ru/perxis/perxis-go/proto/items;items'
+  _ITEM_TRANSLATIONSENTRY._options = None
+  _ITEM_TRANSLATIONSENTRY._serialized_options = b'8\001'
+  _ERROR._serialized_start=149
+  _ERROR._serialized_end=188
+  _DECODEERROR._serialized_start=190
+  _DECODEERROR._serialized_end=241
+  _VALIDATIONERROR._serialized_start=243
+  _VALIDATIONERROR._serialized_end=298
+  _MODIFICATIONERROR._serialized_start=300
+  _MODIFICATIONERROR._serialized_end=357
+  _PERMISSIONS._serialized_start=359
+  _PERMISSIONS._serialized_end=462
+  _ITEM._serialized_start=465
+  _ITEM._serialized_end=1289
+  _ITEM_TRANSLATIONSENTRY._serialized_start=1151
+  _ITEM_TRANSLATIONSENTRY._serialized_end=1227
+  _ITEM_STATE._serialized_start=1229
+  _ITEM_STATE._serialized_end=1289
+  _FILTER._serialized_start=1291
+  _FILTER._serialized_end=1352
+  _CREATEOPTIONS._serialized_start=1354
+  _CREATEOPTIONS._serialized_end=1391
+  _FINDOPTIONS._serialized_start=1393
+  _FINDOPTIONS._serialized_end=1513
+  _UPDATEOPTIONS._serialized_start=1515
+  _UPDATEOPTIONS._serialized_end=1552
+  _GETPUBLISHEDOPTIONS._serialized_start=1554
+  _GETPUBLISHEDOPTIONS._serialized_end=1594
+  _DELETEOPTIONS._serialized_start=1596
+  _DELETEOPTIONS._serialized_end=1626
+  _PUBLISHOPTIONS._serialized_start=1628
+  _PUBLISHOPTIONS._serialized_end=1666
+  _FINDPUBLISHEDOPTIONS._serialized_start=1669
+  _FINDPUBLISHEDOPTIONS._serialized_end=1800
+  _FINDARCHIVEDOPTIONS._serialized_start=1802
+  _FINDARCHIVEDOPTIONS._serialized_end=1861
+  _LISTREVISIONSOPTIONS._serialized_start=1863
+  _LISTREVISIONSOPTIONS._serialized_end=1923
+  _CREATEREQUEST._serialized_start=1925
+  _CREATEREQUEST._serialized_end=2022
+  _CREATERESPONSE._serialized_start=2024
+  _CREATERESPONSE._serialized_end=2078
+  _INTROSPECTREQUEST._serialized_start=2080
+  _INTROSPECTREQUEST._serialized_end=2134
+  _INTROSPECTRESPONSE._serialized_start=2137
+  _INTROSPECTRESPONSE._serialized_end=2276
+  _GETREQUEST._serialized_start=2278
+  _GETREQUEST._serialized_end=2364
+  _GETRESPONSE._serialized_start=2366
+  _GETRESPONSE._serialized_end=2414
+  _FINDREQUEST._serialized_start=2417
+  _FINDREQUEST._serialized_end=2571
+  _FINDRESPONSE._serialized_start=2573
+  _FINDRESPONSE._serialized_end=2638
+  _UPDATEREQUEST._serialized_start=2640
+  _UPDATEREQUEST._serialized_end=2737
+  _DELETEREQUEST._serialized_start=2740
+  _DELETEREQUEST._serialized_end=2876
+  _UNDELETEREQUEST._serialized_start=2878
+  _UNDELETEREQUEST._serialized_end=2969
+  _PUBLISHREQUEST._serialized_start=2971
+  _PUBLISHREQUEST._serialized_end=3070
+  _UNPUBLISHREQUEST._serialized_start=3072
+  _UNPUBLISHREQUEST._serialized_end=3125
+  _GETPUBLISHEDREQUEST._serialized_start=3128
+  _GETPUBLISHEDREQUEST._serialized_end=3276
+  _GETPUBLISHEDRESPONSE._serialized_start=3278
+  _GETPUBLISHEDRESPONSE._serialized_end=3335
+  _FINDPUBLISHEDREQUEST._serialized_start=3338
+  _FINDPUBLISHEDREQUEST._serialized_end=3510
+  _FINDPUBLISHEDRESPONSE._serialized_start=3512
+  _FINDPUBLISHEDRESPONSE._serialized_end=3586
+  _GETREVISIONREQUEST._serialized_start=3588
+  _GETREVISIONREQUEST._serialized_end=3703
+  _GETREVISIONRESPONSE._serialized_start=3705
+  _GETREVISIONRESPONSE._serialized_end=3761
+  _LISTREVISIONSREQUEST._serialized_start=3764
+  _LISTREVISIONSREQUEST._serialized_end=3914
+  _LISTREVISIONSRESPONSE._serialized_start=3916
+  _LISTREVISIONSRESPONSE._serialized_end=3975
+  _ARCHIVEREQUEST._serialized_start=3977
+  _ARCHIVEREQUEST._serialized_end=4028
+  _UNARCHIVEREQUEST._serialized_start=4030
+  _UNARCHIVEREQUEST._serialized_end=4083
+  _FINDARCHIVEDREQUEST._serialized_start=4086
+  _FINDARCHIVEDREQUEST._serialized_end=4256
+  _FINDARCHIVEDRESPONSE._serialized_start=4258
+  _FINDARCHIVEDRESPONSE._serialized_end=4331
+  _ITEMS._serialized_start=4334
+  _ITEMS._serialized_end=5570
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/items/items_pb2_grpc.py b/clients/items/items_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..50b968c82948c33fd0ba345fb13bf7f31ad41b5a
--- /dev/null
+++ b/clients/items/items_pb2_grpc.py
@@ -0,0 +1,576 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from items import items_pb2 as items_dot_items__pb2
+
+
+class ItemsStub(object):
+    """*
+    Сервис API элементов
+    """
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.items.Items/Create',
+                request_serializer=items_dot_items__pb2.CreateRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.CreateResponse.FromString,
+                )
+        self.Introspect = channel.unary_unary(
+                '/content.items.Items/Introspect',
+                request_serializer=items_dot_items__pb2.IntrospectRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.IntrospectResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.items.Items/Get',
+                request_serializer=items_dot_items__pb2.GetRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.GetResponse.FromString,
+                )
+        self.Find = channel.unary_unary(
+                '/content.items.Items/Find',
+                request_serializer=items_dot_items__pb2.FindRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.FindResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.items.Items/Update',
+                request_serializer=items_dot_items__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.items.Items/Delete',
+                request_serializer=items_dot_items__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Undelete = channel.unary_unary(
+                '/content.items.Items/Undelete',
+                request_serializer=items_dot_items__pb2.UndeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Publish = channel.unary_unary(
+                '/content.items.Items/Publish',
+                request_serializer=items_dot_items__pb2.PublishRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Unpublish = channel.unary_unary(
+                '/content.items.Items/Unpublish',
+                request_serializer=items_dot_items__pb2.UnpublishRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.GetPublished = channel.unary_unary(
+                '/content.items.Items/GetPublished',
+                request_serializer=items_dot_items__pb2.GetPublishedRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.GetPublishedResponse.FromString,
+                )
+        self.FindPublished = channel.unary_unary(
+                '/content.items.Items/FindPublished',
+                request_serializer=items_dot_items__pb2.FindPublishedRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.FindPublishedResponse.FromString,
+                )
+        self.GetRevision = channel.unary_unary(
+                '/content.items.Items/GetRevision',
+                request_serializer=items_dot_items__pb2.GetRevisionRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.GetRevisionResponse.FromString,
+                )
+        self.ListRevisions = channel.unary_unary(
+                '/content.items.Items/ListRevisions',
+                request_serializer=items_dot_items__pb2.ListRevisionsRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.ListRevisionsResponse.FromString,
+                )
+        self.Archive = channel.unary_unary(
+                '/content.items.Items/Archive',
+                request_serializer=items_dot_items__pb2.ArchiveRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.FindArchived = channel.unary_unary(
+                '/content.items.Items/FindArchived',
+                request_serializer=items_dot_items__pb2.FindArchivedRequest.SerializeToString,
+                response_deserializer=items_dot_items__pb2.FindArchivedResponse.FromString,
+                )
+        self.Unarchive = channel.unary_unary(
+                '/content.items.Items/Unarchive',
+                request_serializer=items_dot_items__pb2.UnarchiveRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class ItemsServicer(object):
+    """*
+    Сервис API элементов
+    """
+
+    def Create(self, request, context):
+        """*
+        Создать запись
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Introspect(self, request, context):
+        """*
+        Валидация данных записи
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """*
+        Получение записи по идентификатору
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Find(self, request, context):
+        """*
+        Поиск по текущим записям
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Undelete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Publish(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Unpublish(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetPublished(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def FindPublished(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetRevision(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListRevisions(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Archive(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def FindArchived(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Unarchive(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ItemsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=items_dot_items__pb2.CreateRequest.FromString,
+                    response_serializer=items_dot_items__pb2.CreateResponse.SerializeToString,
+            ),
+            'Introspect': grpc.unary_unary_rpc_method_handler(
+                    servicer.Introspect,
+                    request_deserializer=items_dot_items__pb2.IntrospectRequest.FromString,
+                    response_serializer=items_dot_items__pb2.IntrospectResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=items_dot_items__pb2.GetRequest.FromString,
+                    response_serializer=items_dot_items__pb2.GetResponse.SerializeToString,
+            ),
+            'Find': grpc.unary_unary_rpc_method_handler(
+                    servicer.Find,
+                    request_deserializer=items_dot_items__pb2.FindRequest.FromString,
+                    response_serializer=items_dot_items__pb2.FindResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=items_dot_items__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=items_dot_items__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Undelete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Undelete,
+                    request_deserializer=items_dot_items__pb2.UndeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Publish': grpc.unary_unary_rpc_method_handler(
+                    servicer.Publish,
+                    request_deserializer=items_dot_items__pb2.PublishRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Unpublish': grpc.unary_unary_rpc_method_handler(
+                    servicer.Unpublish,
+                    request_deserializer=items_dot_items__pb2.UnpublishRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'GetPublished': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetPublished,
+                    request_deserializer=items_dot_items__pb2.GetPublishedRequest.FromString,
+                    response_serializer=items_dot_items__pb2.GetPublishedResponse.SerializeToString,
+            ),
+            'FindPublished': grpc.unary_unary_rpc_method_handler(
+                    servicer.FindPublished,
+                    request_deserializer=items_dot_items__pb2.FindPublishedRequest.FromString,
+                    response_serializer=items_dot_items__pb2.FindPublishedResponse.SerializeToString,
+            ),
+            'GetRevision': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetRevision,
+                    request_deserializer=items_dot_items__pb2.GetRevisionRequest.FromString,
+                    response_serializer=items_dot_items__pb2.GetRevisionResponse.SerializeToString,
+            ),
+            'ListRevisions': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListRevisions,
+                    request_deserializer=items_dot_items__pb2.ListRevisionsRequest.FromString,
+                    response_serializer=items_dot_items__pb2.ListRevisionsResponse.SerializeToString,
+            ),
+            'Archive': grpc.unary_unary_rpc_method_handler(
+                    servicer.Archive,
+                    request_deserializer=items_dot_items__pb2.ArchiveRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'FindArchived': grpc.unary_unary_rpc_method_handler(
+                    servicer.FindArchived,
+                    request_deserializer=items_dot_items__pb2.FindArchivedRequest.FromString,
+                    response_serializer=items_dot_items__pb2.FindArchivedResponse.SerializeToString,
+            ),
+            'Unarchive': grpc.unary_unary_rpc_method_handler(
+                    servicer.Unarchive,
+                    request_deserializer=items_dot_items__pb2.UnarchiveRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.items.Items', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Items(object):
+    """*
+    Сервис API элементов
+    """
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Create',
+            items_dot_items__pb2.CreateRequest.SerializeToString,
+            items_dot_items__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Introspect(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Introspect',
+            items_dot_items__pb2.IntrospectRequest.SerializeToString,
+            items_dot_items__pb2.IntrospectResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Get',
+            items_dot_items__pb2.GetRequest.SerializeToString,
+            items_dot_items__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Find(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Find',
+            items_dot_items__pb2.FindRequest.SerializeToString,
+            items_dot_items__pb2.FindResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Update',
+            items_dot_items__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Delete',
+            items_dot_items__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Undelete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Undelete',
+            items_dot_items__pb2.UndeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Publish(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Publish',
+            items_dot_items__pb2.PublishRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Unpublish(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Unpublish',
+            items_dot_items__pb2.UnpublishRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetPublished(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/GetPublished',
+            items_dot_items__pb2.GetPublishedRequest.SerializeToString,
+            items_dot_items__pb2.GetPublishedResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def FindPublished(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/FindPublished',
+            items_dot_items__pb2.FindPublishedRequest.SerializeToString,
+            items_dot_items__pb2.FindPublishedResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetRevision(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/GetRevision',
+            items_dot_items__pb2.GetRevisionRequest.SerializeToString,
+            items_dot_items__pb2.GetRevisionResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListRevisions(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/ListRevisions',
+            items_dot_items__pb2.ListRevisionsRequest.SerializeToString,
+            items_dot_items__pb2.ListRevisionsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Archive(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Archive',
+            items_dot_items__pb2.ArchiveRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def FindArchived(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/FindArchived',
+            items_dot_items__pb2.FindArchivedRequest.SerializeToString,
+            items_dot_items__pb2.FindArchivedResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Unarchive(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.items.Items/Unarchive',
+            items_dot_items__pb2.UnarchiveRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/locales/locales_pb2.py b/clients/locales/locales_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..822a5ff23813ece634bda8673436861632530481
--- /dev/null
+++ b/clients/locales/locales_pb2.py
@@ -0,0 +1,89 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: locales/locales.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x15locales/locales.proto\x12\x0f\x63ontent.locales\x1a\x1bgoogle/protobuf/empty.proto\"4\n\x06Locale\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x0c\n\x04name\x18\x03 \x01(\t\"8\n\rCreateRequest\x12\'\n\x06locale\x18\x01 \x01(\x0b\x32\x17.content.locales.Locale\"9\n\x0e\x43reateResponse\x12\'\n\x06locale\x18\x01 \x01(\x0b\x32\x17.content.locales.Locale\"\x1f\n\x0bListRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"8\n\x0cListResponse\x12(\n\x07locales\x18\x01 \x03(\x0b\x32\x17.content.locales.Locale\"4\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x11\n\tlocale_id\x18\x02 \x01(\t2\xe1\x01\n\x07Locales\x12K\n\x06\x43reate\x12\x1e.content.locales.CreateRequest\x1a\x1f.content.locales.CreateResponse\"\x00\x12\x45\n\x04List\x12\x1c.content.locales.ListRequest\x1a\x1d.content.locales.ListResponse\"\x00\x12\x42\n\x06\x44\x65lete\x12\x1e.content.locales.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x34Z2git.perx.ru/perxis/perxis-go/proto/locales;localesb\x06proto3')
+
+
+
+_LOCALE = DESCRIPTOR.message_types_by_name['Locale']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+Locale = _reflection.GeneratedProtocolMessageType('Locale', (_message.Message,), {
+  'DESCRIPTOR' : _LOCALE,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.Locale)
+  })
+_sym_db.RegisterMessage(Locale)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'locales.locales_pb2'
+  # @@protoc_insertion_point(class_scope:content.locales.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+_LOCALES = DESCRIPTOR.services_by_name['Locales']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z2git.perx.ru/perxis/perxis-go/proto/locales;locales'
+  _LOCALE._serialized_start=71
+  _LOCALE._serialized_end=123
+  _CREATEREQUEST._serialized_start=125
+  _CREATEREQUEST._serialized_end=181
+  _CREATERESPONSE._serialized_start=183
+  _CREATERESPONSE._serialized_end=240
+  _LISTREQUEST._serialized_start=242
+  _LISTREQUEST._serialized_end=273
+  _LISTRESPONSE._serialized_start=275
+  _LISTRESPONSE._serialized_end=331
+  _DELETEREQUEST._serialized_start=333
+  _DELETEREQUEST._serialized_end=385
+  _LOCALES._serialized_start=388
+  _LOCALES._serialized_end=613
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/locales/locales_pb2_grpc.py b/clients/locales/locales_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..4a75d07d755194acba32ebeb88b93afed280bf72
--- /dev/null
+++ b/clients/locales/locales_pb2_grpc.py
@@ -0,0 +1,133 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from locales import locales_pb2 as locales_dot_locales__pb2
+
+
+class LocalesStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.locales.Locales/Create',
+                request_serializer=locales_dot_locales__pb2.CreateRequest.SerializeToString,
+                response_deserializer=locales_dot_locales__pb2.CreateResponse.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.locales.Locales/List',
+                request_serializer=locales_dot_locales__pb2.ListRequest.SerializeToString,
+                response_deserializer=locales_dot_locales__pb2.ListResponse.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.locales.Locales/Delete',
+                request_serializer=locales_dot_locales__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class LocalesServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_LocalesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=locales_dot_locales__pb2.CreateRequest.FromString,
+                    response_serializer=locales_dot_locales__pb2.CreateResponse.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=locales_dot_locales__pb2.ListRequest.FromString,
+                    response_serializer=locales_dot_locales__pb2.ListResponse.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=locales_dot_locales__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.locales.Locales', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Locales(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.locales.Locales/Create',
+            locales_dot_locales__pb2.CreateRequest.SerializeToString,
+            locales_dot_locales__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.locales.Locales/List',
+            locales_dot_locales__pb2.ListRequest.SerializeToString,
+            locales_dot_locales__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.locales.Locales/Delete',
+            locales_dot_locales__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/members/members_pb2.py b/clients/members/members_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..93e6a8e091c45bc628dbd457c9ef1dd3273e71d8
--- /dev/null
+++ b/clients/members/members_pb2.py
@@ -0,0 +1,152 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: members/members.proto
+"""Generated protocol buffer code."""
+from google.protobuf.internal import enum_type_wrapper
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x15members/members.proto\x12\x0f\x61\x63\x63ount.members\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"N\n\x06Member\x12\x0e\n\x06org_id\x18\x01 \x01(\t\x12\x0f\n\x07user_id\x18\x02 \x01(\t\x12#\n\x04role\x18\x03 \x01(\x0e\x32\x15.account.members.Role\"R\n\nSetRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\x12\x0f\n\x07user_id\x18\x02 \x01(\t\x12#\n\x04role\x18\x03 \x01(\x0e\x32\x15.account.members.Role\"-\n\nGetRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\x12\x0f\n\x07user_id\x18\x02 \x01(\t\"2\n\x0bGetResponse\x12#\n\x04role\x18\x03 \x01(\x0e\x32\x15.account.members.Role\"0\n\rRemoveRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\x12\x0f\n\x07user_id\x18\x02 \x01(\t\"$\n\x12ListMembersRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\"?\n\x13ListMembersResponse\x12(\n\x07members\x18\x01 \x03(\x0b\x32\x17.account.members.Member\"+\n\x18ListOrganizationsRequest\x12\x0f\n\x07user_id\x18\x01 \x01(\t\"K\n\x19ListOrganizationsResponse\x12.\n\rorganizations\x18\x01 \x03(\x0b\x32\x17.account.members.Member\"F\n\x18OnCollaboratorSetRequest\x12*\n\x0c\x63ollaborator\x18\x01 \x01(\x0b\x32\x14.common.Collaborator\"4\n\x19OnCollaboratorSetResponse\x12\x17\n\x0f\x64\x65layed_task_id\x18\x01 \x01(\t*8\n\x04Role\x12\x0e\n\nNOT_MEMBER\x10\x00\x12\n\n\x06MEMBER\x10\x01\x12\t\n\x05OWNER\x10\x02\x12\t\n\x05\x41\x44MIN\x10\x03\x32\x99\x03\n\x07Members\x12<\n\x03Set\x12\x1b.account.members.SetRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x42\n\x03Get\x12\x1b.account.members.GetRequest\x1a\x1c.account.members.GetResponse\"\x00\x12\x42\n\x06Remove\x12\x1e.account.members.RemoveRequest\x1a\x16.google.protobuf.Empty\"\x00\x12Z\n\x0bListMembers\x12#.account.members.ListMembersRequest\x1a$.account.members.ListMembersResponse\"\x00\x12l\n\x11ListOrganizations\x12).account.members.ListOrganizationsRequest\x1a*.account.members.ListOrganizationsResponse\"\x00\x32x\n\x08Observer\x12l\n\x11OnCollaboratorSet\x12).account.members.OnCollaboratorSetRequest\x1a*.account.members.OnCollaboratorSetResponse\"\x00\x42\x34Z2git.perx.ru/perxis/perxis-go/proto/members;membersb\x06proto3')
+
+_ROLE = DESCRIPTOR.enum_types_by_name['Role']
+Role = enum_type_wrapper.EnumTypeWrapper(_ROLE)
+NOT_MEMBER = 0
+MEMBER = 1
+OWNER = 2
+ADMIN = 3
+
+
+_MEMBER = DESCRIPTOR.message_types_by_name['Member']
+_SETREQUEST = DESCRIPTOR.message_types_by_name['SetRequest']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_REMOVEREQUEST = DESCRIPTOR.message_types_by_name['RemoveRequest']
+_LISTMEMBERSREQUEST = DESCRIPTOR.message_types_by_name['ListMembersRequest']
+_LISTMEMBERSRESPONSE = DESCRIPTOR.message_types_by_name['ListMembersResponse']
+_LISTORGANIZATIONSREQUEST = DESCRIPTOR.message_types_by_name['ListOrganizationsRequest']
+_LISTORGANIZATIONSRESPONSE = DESCRIPTOR.message_types_by_name['ListOrganizationsResponse']
+_ONCOLLABORATORSETREQUEST = DESCRIPTOR.message_types_by_name['OnCollaboratorSetRequest']
+_ONCOLLABORATORSETRESPONSE = DESCRIPTOR.message_types_by_name['OnCollaboratorSetResponse']
+Member = _reflection.GeneratedProtocolMessageType('Member', (_message.Message,), {
+  'DESCRIPTOR' : _MEMBER,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.Member)
+  })
+_sym_db.RegisterMessage(Member)
+
+SetRequest = _reflection.GeneratedProtocolMessageType('SetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _SETREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.SetRequest)
+  })
+_sym_db.RegisterMessage(SetRequest)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+RemoveRequest = _reflection.GeneratedProtocolMessageType('RemoveRequest', (_message.Message,), {
+  'DESCRIPTOR' : _REMOVEREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.RemoveRequest)
+  })
+_sym_db.RegisterMessage(RemoveRequest)
+
+ListMembersRequest = _reflection.GeneratedProtocolMessageType('ListMembersRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTMEMBERSREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.ListMembersRequest)
+  })
+_sym_db.RegisterMessage(ListMembersRequest)
+
+ListMembersResponse = _reflection.GeneratedProtocolMessageType('ListMembersResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTMEMBERSRESPONSE,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.ListMembersResponse)
+  })
+_sym_db.RegisterMessage(ListMembersResponse)
+
+ListOrganizationsRequest = _reflection.GeneratedProtocolMessageType('ListOrganizationsRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTORGANIZATIONSREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.ListOrganizationsRequest)
+  })
+_sym_db.RegisterMessage(ListOrganizationsRequest)
+
+ListOrganizationsResponse = _reflection.GeneratedProtocolMessageType('ListOrganizationsResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTORGANIZATIONSRESPONSE,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.ListOrganizationsResponse)
+  })
+_sym_db.RegisterMessage(ListOrganizationsResponse)
+
+OnCollaboratorSetRequest = _reflection.GeneratedProtocolMessageType('OnCollaboratorSetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _ONCOLLABORATORSETREQUEST,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.OnCollaboratorSetRequest)
+  })
+_sym_db.RegisterMessage(OnCollaboratorSetRequest)
+
+OnCollaboratorSetResponse = _reflection.GeneratedProtocolMessageType('OnCollaboratorSetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _ONCOLLABORATORSETRESPONSE,
+  '__module__' : 'members.members_pb2'
+  # @@protoc_insertion_point(class_scope:account.members.OnCollaboratorSetResponse)
+  })
+_sym_db.RegisterMessage(OnCollaboratorSetResponse)
+
+_MEMBERS = DESCRIPTOR.services_by_name['Members']
+_OBSERVER = DESCRIPTOR.services_by_name['Observer']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z2git.perx.ru/perxis/perxis-go/proto/members;members'
+  _ROLE._serialized_start=756
+  _ROLE._serialized_end=812
+  _MEMBER._serialized_start=92
+  _MEMBER._serialized_end=170
+  _SETREQUEST._serialized_start=172
+  _SETREQUEST._serialized_end=254
+  _GETREQUEST._serialized_start=256
+  _GETREQUEST._serialized_end=301
+  _GETRESPONSE._serialized_start=303
+  _GETRESPONSE._serialized_end=353
+  _REMOVEREQUEST._serialized_start=355
+  _REMOVEREQUEST._serialized_end=403
+  _LISTMEMBERSREQUEST._serialized_start=405
+  _LISTMEMBERSREQUEST._serialized_end=441
+  _LISTMEMBERSRESPONSE._serialized_start=443
+  _LISTMEMBERSRESPONSE._serialized_end=506
+  _LISTORGANIZATIONSREQUEST._serialized_start=508
+  _LISTORGANIZATIONSREQUEST._serialized_end=551
+  _LISTORGANIZATIONSRESPONSE._serialized_start=553
+  _LISTORGANIZATIONSRESPONSE._serialized_end=628
+  _ONCOLLABORATORSETREQUEST._serialized_start=630
+  _ONCOLLABORATORSETREQUEST._serialized_end=700
+  _ONCOLLABORATORSETRESPONSE._serialized_start=702
+  _ONCOLLABORATORSETRESPONSE._serialized_end=754
+  _MEMBERS._serialized_start=815
+  _MEMBERS._serialized_end=1224
+  _OBSERVER._serialized_start=1226
+  _OBSERVER._serialized_end=1346
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/members/members_pb2_grpc.py b/clients/members/members_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..29a5bb96a1ca24a29c49960111fab11faab1e879
--- /dev/null
+++ b/clients/members/members_pb2_grpc.py
@@ -0,0 +1,260 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from members import members_pb2 as members_dot_members__pb2
+
+
+class MembersStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Set = channel.unary_unary(
+                '/account.members.Members/Set',
+                request_serializer=members_dot_members__pb2.SetRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/account.members.Members/Get',
+                request_serializer=members_dot_members__pb2.GetRequest.SerializeToString,
+                response_deserializer=members_dot_members__pb2.GetResponse.FromString,
+                )
+        self.Remove = channel.unary_unary(
+                '/account.members.Members/Remove',
+                request_serializer=members_dot_members__pb2.RemoveRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.ListMembers = channel.unary_unary(
+                '/account.members.Members/ListMembers',
+                request_serializer=members_dot_members__pb2.ListMembersRequest.SerializeToString,
+                response_deserializer=members_dot_members__pb2.ListMembersResponse.FromString,
+                )
+        self.ListOrganizations = channel.unary_unary(
+                '/account.members.Members/ListOrganizations',
+                request_serializer=members_dot_members__pb2.ListOrganizationsRequest.SerializeToString,
+                response_deserializer=members_dot_members__pb2.ListOrganizationsResponse.FromString,
+                )
+
+
+class MembersServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Set(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Remove(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListMembers(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def ListOrganizations(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_MembersServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Set': grpc.unary_unary_rpc_method_handler(
+                    servicer.Set,
+                    request_deserializer=members_dot_members__pb2.SetRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=members_dot_members__pb2.GetRequest.FromString,
+                    response_serializer=members_dot_members__pb2.GetResponse.SerializeToString,
+            ),
+            'Remove': grpc.unary_unary_rpc_method_handler(
+                    servicer.Remove,
+                    request_deserializer=members_dot_members__pb2.RemoveRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'ListMembers': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListMembers,
+                    request_deserializer=members_dot_members__pb2.ListMembersRequest.FromString,
+                    response_serializer=members_dot_members__pb2.ListMembersResponse.SerializeToString,
+            ),
+            'ListOrganizations': grpc.unary_unary_rpc_method_handler(
+                    servicer.ListOrganizations,
+                    request_deserializer=members_dot_members__pb2.ListOrganizationsRequest.FromString,
+                    response_serializer=members_dot_members__pb2.ListOrganizationsResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'account.members.Members', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Members(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Set(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Members/Set',
+            members_dot_members__pb2.SetRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Members/Get',
+            members_dot_members__pb2.GetRequest.SerializeToString,
+            members_dot_members__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Remove(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Members/Remove',
+            members_dot_members__pb2.RemoveRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListMembers(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Members/ListMembers',
+            members_dot_members__pb2.ListMembersRequest.SerializeToString,
+            members_dot_members__pb2.ListMembersResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def ListOrganizations(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Members/ListOrganizations',
+            members_dot_members__pb2.ListOrganizationsRequest.SerializeToString,
+            members_dot_members__pb2.ListOrganizationsResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+
+class ObserverStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.OnCollaboratorSet = channel.unary_unary(
+                '/account.members.Observer/OnCollaboratorSet',
+                request_serializer=members_dot_members__pb2.OnCollaboratorSetRequest.SerializeToString,
+                response_deserializer=members_dot_members__pb2.OnCollaboratorSetResponse.FromString,
+                )
+
+
+class ObserverServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def OnCollaboratorSet(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ObserverServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'OnCollaboratorSet': grpc.unary_unary_rpc_method_handler(
+                    servicer.OnCollaboratorSet,
+                    request_deserializer=members_dot_members__pb2.OnCollaboratorSetRequest.FromString,
+                    response_serializer=members_dot_members__pb2.OnCollaboratorSetResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'account.members.Observer', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Observer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def OnCollaboratorSet(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.members.Observer/OnCollaboratorSet',
+            members_dot_members__pb2.OnCollaboratorSetRequest.SerializeToString,
+            members_dot_members__pb2.OnCollaboratorSetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/organizations/organizations_pb2.py b/clients/organizations/organizations_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..1f9d8997596b68e953b8d40d36944d8edd1dca28
--- /dev/null
+++ b/clients/organizations/organizations_pb2.py
@@ -0,0 +1,130 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: organizations/organizations.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n!organizations/organizations.proto\x12\x15\x61\x63\x63ount.organizations\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"s\n\x0cOrganization\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x03 \x01(\t\x12\x10\n\x08logo_url\x18\x05 \x01(\t\x12\x15\n\x08owner_id\x18\x06 \x01(\tH\x00\x88\x01\x01\x42\x0b\n\t_owner_id\"A\n\rCreateRequest\x12\x30\n\x03org\x18\x01 \x01(\x0b\x32#.account.organizations.Organization\"F\n\x0e\x43reateResponse\x12\x34\n\x07\x63reated\x18\x01 \x01(\x0b\x32#.account.organizations.Organization\"\x1c\n\nGetRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\"?\n\x0bGetResponse\x12\x30\n\x03org\x18\x01 \x01(\x0b\x32#.account.organizations.Organization\"A\n\rUpdateRequest\x12\x30\n\x03org\x18\x01 \x01(\x0b\x32#.account.organizations.Organization\"\x1f\n\rDeleteRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\"$\n\x06\x46ilter\x12\x0b\n\x03ids\x18\x03 \x03(\t\x12\r\n\x05names\x18\x04 \x03(\t\"_\n\x0b\x46indRequest\x12-\n\x06\x66ilter\x18\x01 \x01(\x0b\x32\x1d.account.organizations.Filter\x12!\n\x04opts\x18\x02 \x01(\x0b\x32\x13.common.FindOptions\"P\n\x0c\x46indResponse\x12\x31\n\x04orgs\x18\x01 \x03(\x0b\x32#.account.organizations.Organization\x12\r\n\x05total\x18\x02 \x01(\x03\x32\x9f\x03\n\rOrganizations\x12W\n\x06\x43reate\x12$.account.organizations.CreateRequest\x1a%.account.organizations.CreateResponse\"\x00\x12N\n\x03Get\x12!.account.organizations.GetRequest\x1a\".account.organizations.GetResponse\"\x00\x12H\n\x06\x44\x65lete\x12$.account.organizations.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x12Q\n\x04\x46ind\x12\".account.organizations.FindRequest\x1a#.account.organizations.FindResponse\"\x00\x12H\n\x06Update\x12$.account.organizations.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x42@Z>git.perx.ru/perxis/perxis-go/proto/organizations;organizationsb\x06proto3')
+
+
+
+_ORGANIZATION = DESCRIPTOR.message_types_by_name['Organization']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_FILTER = DESCRIPTOR.message_types_by_name['Filter']
+_FINDREQUEST = DESCRIPTOR.message_types_by_name['FindRequest']
+_FINDRESPONSE = DESCRIPTOR.message_types_by_name['FindResponse']
+Organization = _reflection.GeneratedProtocolMessageType('Organization', (_message.Message,), {
+  'DESCRIPTOR' : _ORGANIZATION,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.Organization)
+  })
+_sym_db.RegisterMessage(Organization)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+Filter = _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+  'DESCRIPTOR' : _FILTER,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.Filter)
+  })
+_sym_db.RegisterMessage(Filter)
+
+FindRequest = _reflection.GeneratedProtocolMessageType('FindRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDREQUEST,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.FindRequest)
+  })
+_sym_db.RegisterMessage(FindRequest)
+
+FindResponse = _reflection.GeneratedProtocolMessageType('FindResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDRESPONSE,
+  '__module__' : 'organizations.organizations_pb2'
+  # @@protoc_insertion_point(class_scope:account.organizations.FindResponse)
+  })
+_sym_db.RegisterMessage(FindResponse)
+
+_ORGANIZATIONS = DESCRIPTOR.services_by_name['Organizations']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z>git.perx.ru/perxis/perxis-go/proto/organizations;organizations'
+  _ORGANIZATION._serialized_start=110
+  _ORGANIZATION._serialized_end=225
+  _CREATEREQUEST._serialized_start=227
+  _CREATEREQUEST._serialized_end=292
+  _CREATERESPONSE._serialized_start=294
+  _CREATERESPONSE._serialized_end=364
+  _GETREQUEST._serialized_start=366
+  _GETREQUEST._serialized_end=394
+  _GETRESPONSE._serialized_start=396
+  _GETRESPONSE._serialized_end=459
+  _UPDATEREQUEST._serialized_start=461
+  _UPDATEREQUEST._serialized_end=526
+  _DELETEREQUEST._serialized_start=528
+  _DELETEREQUEST._serialized_end=559
+  _FILTER._serialized_start=561
+  _FILTER._serialized_end=597
+  _FINDREQUEST._serialized_start=599
+  _FINDREQUEST._serialized_end=694
+  _FINDRESPONSE._serialized_start=696
+  _FINDRESPONSE._serialized_end=776
+  _ORGANIZATIONS._serialized_start=779
+  _ORGANIZATIONS._serialized_end=1194
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/organizations/organizations_pb2_grpc.py b/clients/organizations/organizations_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..24616c51b229d62ab557aa73cd72ff4bf2e1c5a4
--- /dev/null
+++ b/clients/organizations/organizations_pb2_grpc.py
@@ -0,0 +1,199 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from organizations import organizations_pb2 as organizations_dot_organizations__pb2
+
+
+class OrganizationsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/account.organizations.Organizations/Create',
+                request_serializer=organizations_dot_organizations__pb2.CreateRequest.SerializeToString,
+                response_deserializer=organizations_dot_organizations__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/account.organizations.Organizations/Get',
+                request_serializer=organizations_dot_organizations__pb2.GetRequest.SerializeToString,
+                response_deserializer=organizations_dot_organizations__pb2.GetResponse.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/account.organizations.Organizations/Delete',
+                request_serializer=organizations_dot_organizations__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Find = channel.unary_unary(
+                '/account.organizations.Organizations/Find',
+                request_serializer=organizations_dot_organizations__pb2.FindRequest.SerializeToString,
+                response_deserializer=organizations_dot_organizations__pb2.FindResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/account.organizations.Organizations/Update',
+                request_serializer=organizations_dot_organizations__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class OrganizationsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Find(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_OrganizationsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=organizations_dot_organizations__pb2.CreateRequest.FromString,
+                    response_serializer=organizations_dot_organizations__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=organizations_dot_organizations__pb2.GetRequest.FromString,
+                    response_serializer=organizations_dot_organizations__pb2.GetResponse.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=organizations_dot_organizations__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Find': grpc.unary_unary_rpc_method_handler(
+                    servicer.Find,
+                    request_deserializer=organizations_dot_organizations__pb2.FindRequest.FromString,
+                    response_serializer=organizations_dot_organizations__pb2.FindResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=organizations_dot_organizations__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'account.organizations.Organizations', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Organizations(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.organizations.Organizations/Create',
+            organizations_dot_organizations__pb2.CreateRequest.SerializeToString,
+            organizations_dot_organizations__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.organizations.Organizations/Get',
+            organizations_dot_organizations__pb2.GetRequest.SerializeToString,
+            organizations_dot_organizations__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.organizations.Organizations/Delete',
+            organizations_dot_organizations__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Find(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.organizations.Organizations/Find',
+            organizations_dot_organizations__pb2.FindRequest.SerializeToString,
+            organizations_dot_organizations__pb2.FindResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.organizations.Organizations/Update',
+            organizations_dot_organizations__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/references/references_pb2.py b/clients/references/references_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..12f7e4649c89d6460e4219541364f4f25025b46a
--- /dev/null
+++ b/clients/references/references_pb2.py
@@ -0,0 +1,59 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: references/references.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from items import items_pb2 as items_dot_items__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1breferences/references.proto\x12\x12\x63ontent.references\x1a\x11items/items.proto\".\n\tReference\x12\n\n\x02id\x18\x01 \x01(\t\x12\x15\n\rcollection_id\x18\x02 \x01(\t\"a\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0e\n\x06\x65nv_id\x18\x02 \x01(\t\x12\x31\n\nreferences\x18\x03 \x03(\x0b\x32\x1d.content.references.Reference\"b\n\x0bGetResponse\x12\"\n\x05items\x18\x01 \x03(\x0b\x32\x13.content.items.Item\x12/\n\x08notfound\x18\x02 \x03(\x0b\x32\x1d.content.references.Reference2V\n\nReferences\x12H\n\x03Get\x12\x1e.content.references.GetRequest\x1a\x1f.content.references.GetResponse\"\x00\x42:Z8git.perx.ru/perxis/perxis-go/proto/references;referencesb\x06proto3')
+
+
+
+_REFERENCE = DESCRIPTOR.message_types_by_name['Reference']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+Reference = _reflection.GeneratedProtocolMessageType('Reference', (_message.Message,), {
+  'DESCRIPTOR' : _REFERENCE,
+  '__module__' : 'references.references_pb2'
+  # @@protoc_insertion_point(class_scope:content.references.Reference)
+  })
+_sym_db.RegisterMessage(Reference)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'references.references_pb2'
+  # @@protoc_insertion_point(class_scope:content.references.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'references.references_pb2'
+  # @@protoc_insertion_point(class_scope:content.references.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+_REFERENCES = DESCRIPTOR.services_by_name['References']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z8git.perx.ru/perxis/perxis-go/proto/references;references'
+  _REFERENCE._serialized_start=70
+  _REFERENCE._serialized_end=116
+  _GETREQUEST._serialized_start=118
+  _GETREQUEST._serialized_end=215
+  _GETRESPONSE._serialized_start=217
+  _GETRESPONSE._serialized_end=315
+  _REFERENCES._serialized_start=317
+  _REFERENCES._serialized_end=403
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/references/references_pb2_grpc.py b/clients/references/references_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..fa813976bd3686090c4fa040bba92075678dd70f
--- /dev/null
+++ b/clients/references/references_pb2_grpc.py
@@ -0,0 +1,72 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from references import references_pb2 as references_dot_references__pb2
+
+
+class ReferencesStub(object):
+    """References принимает список ссылок на записи и возвращает два списка:
+    список найденных записей и не найденных ссылок
+    """
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Get = channel.unary_unary(
+                '/content.references.References/Get',
+                request_serializer=references_dot_references__pb2.GetRequest.SerializeToString,
+                response_deserializer=references_dot_references__pb2.GetResponse.FromString,
+                )
+
+
+class ReferencesServicer(object):
+    """References принимает список ссылок на записи и возвращает два списка:
+    список найденных записей и не найденных ссылок
+    """
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_ReferencesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=references_dot_references__pb2.GetRequest.FromString,
+                    response_serializer=references_dot_references__pb2.GetResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.references.References', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class References(object):
+    """References принимает список ссылок на записи и возвращает два списка:
+    список найденных записей и не найденных ссылок
+    """
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.references.References/Get',
+            references_dot_references__pb2.GetRequest.SerializeToString,
+            references_dot_references__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/roles/roles_pb2.py b/clients/roles/roles_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..d602286ac968171ef358def7452b7e7b61b9c6a4
--- /dev/null
+++ b/clients/roles/roles_pb2.py
@@ -0,0 +1,120 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: roles/roles.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x11roles/roles.proto\x12\rcontent.roles\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"\x86\x01\n\x04Role\x12\n\n\x02id\x18\x01 \x01(\t\x12\x10\n\x08space_id\x18\x02 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x03 \x01(\t\x12\x1b\n\x05rules\x18\x04 \x03(\x0b\x32\x0c.common.Rule\x12\x14\n\x0c\x65nvironments\x18\x05 \x03(\t\x12\x18\n\x10\x61llow_management\x18\n \x01(\x08\"2\n\rCreateRequest\x12!\n\x04role\x18\x01 \x01(\x0b\x32\x13.content.roles.Role\"6\n\x0e\x43reateResponse\x12$\n\x07\x63reated\x18\x01 \x01(\x0b\x32\x13.content.roles.Role\"/\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07role_id\x18\x02 \x01(\t\"0\n\x0bGetResponse\x12!\n\x04role\x18\x01 \x01(\x0b\x32\x13.content.roles.Role\"2\n\rUpdateRequest\x12!\n\x04role\x18\x01 \x01(\x0b\x32\x13.content.roles.Role\"\x1f\n\x0bListRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"2\n\x0cListResponse\x12\"\n\x05roles\x18\x01 \x03(\x0b\x32\x13.content.roles.Role\"2\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12\x0f\n\x07role_id\x18\x02 \x01(\t2\xd7\x02\n\x05Roles\x12G\n\x06\x43reate\x12\x1c.content.roles.CreateRequest\x1a\x1d.content.roles.CreateResponse\"\x00\x12>\n\x03Get\x12\x19.content.roles.GetRequest\x1a\x1a.content.roles.GetResponse\"\x00\x12@\n\x06Update\x12\x1c.content.roles.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x41\n\x04List\x12\x1a.content.roles.ListRequest\x1a\x1b.content.roles.ListResponse\"\x00\x12@\n\x06\x44\x65lete\x12\x1c.content.roles.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x30Z.git.perx.ru/perxis/perxis-go/proto/roles;rolesb\x06proto3')
+
+
+
+_ROLE = DESCRIPTOR.message_types_by_name['Role']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+Role = _reflection.GeneratedProtocolMessageType('Role', (_message.Message,), {
+  'DESCRIPTOR' : _ROLE,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.Role)
+  })
+_sym_db.RegisterMessage(Role)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'roles.roles_pb2'
+  # @@protoc_insertion_point(class_scope:content.roles.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+_ROLES = DESCRIPTOR.services_by_name['Roles']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z.git.perx.ru/perxis/perxis-go/proto/roles;roles'
+  _ROLE._serialized_start=87
+  _ROLE._serialized_end=221
+  _CREATEREQUEST._serialized_start=223
+  _CREATEREQUEST._serialized_end=273
+  _CREATERESPONSE._serialized_start=275
+  _CREATERESPONSE._serialized_end=329
+  _GETREQUEST._serialized_start=331
+  _GETREQUEST._serialized_end=378
+  _GETRESPONSE._serialized_start=380
+  _GETRESPONSE._serialized_end=428
+  _UPDATEREQUEST._serialized_start=430
+  _UPDATEREQUEST._serialized_end=480
+  _LISTREQUEST._serialized_start=482
+  _LISTREQUEST._serialized_end=513
+  _LISTRESPONSE._serialized_start=515
+  _LISTRESPONSE._serialized_end=565
+  _DELETEREQUEST._serialized_start=567
+  _DELETEREQUEST._serialized_end=617
+  _ROLES._serialized_start=620
+  _ROLES._serialized_end=963
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/roles/roles_pb2_grpc.py b/clients/roles/roles_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..6b779df97a764a24189c25bd4cd516dcc18581e3
--- /dev/null
+++ b/clients/roles/roles_pb2_grpc.py
@@ -0,0 +1,204 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from roles import roles_pb2 as roles_dot_roles__pb2
+
+
+class RolesStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.roles.Roles/Create',
+                request_serializer=roles_dot_roles__pb2.CreateRequest.SerializeToString,
+                response_deserializer=roles_dot_roles__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.roles.Roles/Get',
+                request_serializer=roles_dot_roles__pb2.GetRequest.SerializeToString,
+                response_deserializer=roles_dot_roles__pb2.GetResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.roles.Roles/Update',
+                request_serializer=roles_dot_roles__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.roles.Roles/List',
+                request_serializer=roles_dot_roles__pb2.ListRequest.SerializeToString,
+                response_deserializer=roles_dot_roles__pb2.ListResponse.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.roles.Roles/Delete',
+                request_serializer=roles_dot_roles__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class RolesServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Create - создает роль в рамках пространства
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Get - возвращает роль по id
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Update - обновляет параметры роли
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """List - возвращает список ролей, созданных в пространстве
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Delete - удаляет указанную роль из пространстве
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_RolesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=roles_dot_roles__pb2.CreateRequest.FromString,
+                    response_serializer=roles_dot_roles__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=roles_dot_roles__pb2.GetRequest.FromString,
+                    response_serializer=roles_dot_roles__pb2.GetResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=roles_dot_roles__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=roles_dot_roles__pb2.ListRequest.FromString,
+                    response_serializer=roles_dot_roles__pb2.ListResponse.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=roles_dot_roles__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.roles.Roles', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Roles(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.roles.Roles/Create',
+            roles_dot_roles__pb2.CreateRequest.SerializeToString,
+            roles_dot_roles__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.roles.Roles/Get',
+            roles_dot_roles__pb2.GetRequest.SerializeToString,
+            roles_dot_roles__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.roles.Roles/Update',
+            roles_dot_roles__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.roles.Roles/List',
+            roles_dot_roles__pb2.ListRequest.SerializeToString,
+            roles_dot_roles__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.roles.Roles/Delete',
+            roles_dot_roles__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/spaces/spaces_pb2.py b/clients/spaces/spaces_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..6d9f0e6249f59493025e80b6f08ff267eeeae2df
--- /dev/null
+++ b/clients/spaces/spaces_pb2.py
@@ -0,0 +1,152 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: spaces/spaces.proto
+"""Generated protocol buffer code."""
+from google.protobuf.internal import enum_type_wrapper
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x13spaces/spaces.proto\x12\x0e\x63ontent.spaces\x1a\x1bgoogle/protobuf/empty.proto\"\x94\x01\n\x05Space\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0e\n\x06org_id\x18\x02 \x01(\t\x12\x0c\n\x04name\x18\x03 \x01(\t\x12\x13\n\x0b\x64\x65scription\x18\x04 \x01(\t\x12$\n\x05state\x18\x05 \x01(\x0e\x32\x15.content.spaces.State\x12&\n\x06\x63onfig\x18\n \x01(\x0b\x32\x16.content.spaces.Config\"\x1a\n\x06\x43onfig\x12\x10\n\x08\x66\x65\x61tures\x18\x01 \x03(\t\"5\n\rCreateRequest\x12$\n\x05space\x18\x01 \x01(\x0b\x32\x15.content.spaces.Space\"8\n\x0e\x43reateResponse\x12&\n\x07\x63reated\x18\x01 \x01(\x0b\x32\x15.content.spaces.Space\"\x1e\n\nGetRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\"3\n\x0bGetResponse\x12$\n\x05space\x18\x01 \x01(\x0b\x32\x15.content.spaces.Space\"\x1d\n\x0bListRequest\x12\x0e\n\x06org_id\x18\x01 \x01(\t\"5\n\x0cListResponse\x12%\n\x06spaces\x18\x01 \x03(\x0b\x32\x15.content.spaces.Space\"5\n\rUpdateRequest\x12$\n\x05space\x18\x01 \x01(\x0b\x32\x15.content.spaces.Space\"O\n\x13UpdateConfigRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t\x12&\n\x06\x63onfig\x18\x02 \x01(\x0b\x32\x16.content.spaces.Config\"!\n\rDeleteRequest\x12\x10\n\x08space_id\x18\x01 \x01(\t*p\n\x05State\x12\x0b\n\x07UNKNOWN\x10\x00\x12\x07\n\x03NEW\x10\x01\x12\t\n\x05READY\x10\x02\x12\r\n\tPREPARING\x10\x03\x12\x0f\n\x0bMAINTENANCE\x10\x04\x12\r\n\tMIGRATION\x10\x05\x12\x0c\n\x08\x44\x45LETING\x10\x06\x12\t\n\x05\x45RROR\x10\x07\x32\xaf\x03\n\x06Spaces\x12I\n\x06\x43reate\x12\x1d.content.spaces.CreateRequest\x1a\x1e.content.spaces.CreateResponse\"\x00\x12@\n\x03Get\x12\x1a.content.spaces.GetRequest\x1a\x1b.content.spaces.GetResponse\"\x00\x12\x43\n\x04List\x12\x1b.content.spaces.ListRequest\x1a\x1c.content.spaces.ListResponse\"\x00\x12\x41\n\x06Update\x12\x1d.content.spaces.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12M\n\x0cUpdateConfig\x12#.content.spaces.UpdateConfigRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\x41\n\x06\x44\x65lete\x12\x1d.content.spaces.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x42\x32Z0git.perx.ru/perxis/perxis-go/proto/spaces;spacesb\x06proto3')
+
+_STATE = DESCRIPTOR.enum_types_by_name['State']
+State = enum_type_wrapper.EnumTypeWrapper(_STATE)
+UNKNOWN = 0
+NEW = 1
+READY = 2
+PREPARING = 3
+MAINTENANCE = 4
+MIGRATION = 5
+DELETING = 6
+ERROR = 7
+
+
+_SPACE = DESCRIPTOR.message_types_by_name['Space']
+_CONFIG = DESCRIPTOR.message_types_by_name['Config']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_LISTREQUEST = DESCRIPTOR.message_types_by_name['ListRequest']
+_LISTRESPONSE = DESCRIPTOR.message_types_by_name['ListResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_UPDATECONFIGREQUEST = DESCRIPTOR.message_types_by_name['UpdateConfigRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+Space = _reflection.GeneratedProtocolMessageType('Space', (_message.Message,), {
+  'DESCRIPTOR' : _SPACE,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.Space)
+  })
+_sym_db.RegisterMessage(Space)
+
+Config = _reflection.GeneratedProtocolMessageType('Config', (_message.Message,), {
+  'DESCRIPTOR' : _CONFIG,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.Config)
+  })
+_sym_db.RegisterMessage(Config)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+ListRequest = _reflection.GeneratedProtocolMessageType('ListRequest', (_message.Message,), {
+  'DESCRIPTOR' : _LISTREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.ListRequest)
+  })
+_sym_db.RegisterMessage(ListRequest)
+
+ListResponse = _reflection.GeneratedProtocolMessageType('ListResponse', (_message.Message,), {
+  'DESCRIPTOR' : _LISTRESPONSE,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.ListResponse)
+  })
+_sym_db.RegisterMessage(ListResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+UpdateConfigRequest = _reflection.GeneratedProtocolMessageType('UpdateConfigRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATECONFIGREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.UpdateConfigRequest)
+  })
+_sym_db.RegisterMessage(UpdateConfigRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'spaces.spaces_pb2'
+  # @@protoc_insertion_point(class_scope:content.spaces.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+_SPACES = DESCRIPTOR.services_by_name['Spaces']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z0git.perx.ru/perxis/perxis-go/proto/spaces;spaces'
+  _STATE._serialized_start=702
+  _STATE._serialized_end=814
+  _SPACE._serialized_start=69
+  _SPACE._serialized_end=217
+  _CONFIG._serialized_start=219
+  _CONFIG._serialized_end=245
+  _CREATEREQUEST._serialized_start=247
+  _CREATEREQUEST._serialized_end=300
+  _CREATERESPONSE._serialized_start=302
+  _CREATERESPONSE._serialized_end=358
+  _GETREQUEST._serialized_start=360
+  _GETREQUEST._serialized_end=390
+  _GETRESPONSE._serialized_start=392
+  _GETRESPONSE._serialized_end=443
+  _LISTREQUEST._serialized_start=445
+  _LISTREQUEST._serialized_end=474
+  _LISTRESPONSE._serialized_start=476
+  _LISTRESPONSE._serialized_end=529
+  _UPDATEREQUEST._serialized_start=531
+  _UPDATEREQUEST._serialized_end=584
+  _UPDATECONFIGREQUEST._serialized_start=586
+  _UPDATECONFIGREQUEST._serialized_end=665
+  _DELETEREQUEST._serialized_start=667
+  _DELETEREQUEST._serialized_end=700
+  _SPACES._serialized_start=817
+  _SPACES._serialized_end=1248
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/spaces/spaces_pb2_grpc.py b/clients/spaces/spaces_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..2ab95a7b82f2899207dc7a4ed6c00b3c734db268
--- /dev/null
+++ b/clients/spaces/spaces_pb2_grpc.py
@@ -0,0 +1,232 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from spaces import spaces_pb2 as spaces_dot_spaces__pb2
+
+
+class SpacesStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/content.spaces.Spaces/Create',
+                request_serializer=spaces_dot_spaces__pb2.CreateRequest.SerializeToString,
+                response_deserializer=spaces_dot_spaces__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/content.spaces.Spaces/Get',
+                request_serializer=spaces_dot_spaces__pb2.GetRequest.SerializeToString,
+                response_deserializer=spaces_dot_spaces__pb2.GetResponse.FromString,
+                )
+        self.List = channel.unary_unary(
+                '/content.spaces.Spaces/List',
+                request_serializer=spaces_dot_spaces__pb2.ListRequest.SerializeToString,
+                response_deserializer=spaces_dot_spaces__pb2.ListResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/content.spaces.Spaces/Update',
+                request_serializer=spaces_dot_spaces__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.UpdateConfig = channel.unary_unary(
+                '/content.spaces.Spaces/UpdateConfig',
+                request_serializer=spaces_dot_spaces__pb2.UpdateConfigRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/content.spaces.Spaces/Delete',
+                request_serializer=spaces_dot_spaces__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+
+
+class SpacesServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def List(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def UpdateConfig(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_SpacesServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=spaces_dot_spaces__pb2.CreateRequest.FromString,
+                    response_serializer=spaces_dot_spaces__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=spaces_dot_spaces__pb2.GetRequest.FromString,
+                    response_serializer=spaces_dot_spaces__pb2.GetResponse.SerializeToString,
+            ),
+            'List': grpc.unary_unary_rpc_method_handler(
+                    servicer.List,
+                    request_deserializer=spaces_dot_spaces__pb2.ListRequest.FromString,
+                    response_serializer=spaces_dot_spaces__pb2.ListResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=spaces_dot_spaces__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'UpdateConfig': grpc.unary_unary_rpc_method_handler(
+                    servicer.UpdateConfig,
+                    request_deserializer=spaces_dot_spaces__pb2.UpdateConfigRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=spaces_dot_spaces__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.spaces.Spaces', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Spaces(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/Create',
+            spaces_dot_spaces__pb2.CreateRequest.SerializeToString,
+            spaces_dot_spaces__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/Get',
+            spaces_dot_spaces__pb2.GetRequest.SerializeToString,
+            spaces_dot_spaces__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def List(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/List',
+            spaces_dot_spaces__pb2.ListRequest.SerializeToString,
+            spaces_dot_spaces__pb2.ListResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/Update',
+            spaces_dot_spaces__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def UpdateConfig(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/UpdateConfig',
+            spaces_dot_spaces__pb2.UpdateConfigRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.spaces.Spaces/Delete',
+            spaces_dot_spaces__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/users/users_pb2.py b/clients/users/users_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..f5d126b092a825bcb786ecc60ef60c91f44b9276
--- /dev/null
+++ b/clients/users/users_pb2.py
@@ -0,0 +1,151 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: users/users.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from google.protobuf import wrappers_pb2 as google_dot_protobuf_dot_wrappers__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x11users/users.proto\x12\raccount.users\x1a\x1bgoogle/protobuf/empty.proto\x1a\x1egoogle/protobuf/wrappers.proto\x1a\x13\x63ommon/common.proto\"\xcd\x01\n\x04User\x12\n\n\x02id\x18\x01 \x01(\t\x12\x0c\n\x04name\x18\x02 \x01(\t\x12\x12\n\nidentities\x18\x03 \x03(\t\x12\x14\n\x0c\x64isplay_name\x18\x04 \x01(\t\x12\r\n\x05\x65mail\x18\x05 \x01(\t\x12\x32\n\x0e\x65mail_verified\x18\x06 \x01(\x0b\x32\x1a.google.protobuf.BoolValue\x12\x12\n\navatar_url\x18\x07 \x01(\t\x12*\n\x06system\x18\n \x01(\x0b\x32\x1a.google.protobuf.BoolValue\"\xbb\x01\n\x06\x46ilter\x12\n\n\x02id\x18\x01 \x03(\t\x12\x0c\n\x04name\x18\x02 \x03(\t\x12\x12\n\nidentities\x18\x03 \x03(\t\x12\x14\n\x0c\x64isplay_name\x18\x04 \x03(\t\x12\r\n\x05\x65mail\x18\x05 \x03(\t\x12\x32\n\x0e\x65mail_verified\x18\x06 \x01(\x0b\x32\x1a.google.protobuf.BoolValue\x12*\n\x06system\x18\n \x01(\x0b\x32\x1a.google.protobuf.BoolValue\"4\n\rCreateRequest\x12#\n\x06\x63reate\x18\x01 \x01(\x0b\x32\x13.account.users.User\"3\n\x0e\x43reateResponse\x12!\n\x04user\x18\x01 \x01(\x0b\x32\x13.account.users.User\"\x1d\n\nGetRequest\x12\x0f\n\x07user_id\x18\x01 \x01(\t\"0\n\x0bGetResponse\x12!\n\x04user\x18\x01 \x01(\x0b\x32\x13.account.users.User\"Z\n\x0b\x46indRequest\x12%\n\x06\x66ilter\x18\x01 \x01(\x0b\x32\x15.account.users.Filter\x12$\n\x07options\x18\n \x01(\x0b\x32\x13.common.FindOptions\"A\n\x0c\x46indResponse\x12\"\n\x05users\x18\x01 \x03(\x0b\x32\x13.account.users.User\x12\r\n\x05total\x18\x02 \x01(\x03\"4\n\rUpdateRequest\x12#\n\x06update\x18\x02 \x01(\x0b\x32\x13.account.users.User\" \n\rDeleteRequest\x12\x0f\n\x07user_id\x18\x01 \x01(\t\"(\n\x14GetByIdentityRequest\x12\x10\n\x08identity\x18\x01 \x01(\t\":\n\x15GetByIdentityResponse\x12!\n\x04user\x18\x01 \x01(\x0b\x32\x13.account.users.User2\xb5\x03\n\x05Users\x12G\n\x06\x43reate\x12\x1c.account.users.CreateRequest\x1a\x1d.account.users.CreateResponse\"\x00\x12>\n\x03Get\x12\x19.account.users.GetRequest\x1a\x1a.account.users.GetResponse\"\x00\x12\x41\n\x04\x46ind\x12\x1a.account.users.FindRequest\x1a\x1b.account.users.FindResponse\"\x00\x12@\n\x06Update\x12\x1c.account.users.UpdateRequest\x1a\x16.google.protobuf.Empty\"\x00\x12@\n\x06\x44\x65lete\x12\x1c.account.users.DeleteRequest\x1a\x16.google.protobuf.Empty\"\x00\x12\\\n\rGetByIdentity\x12#.account.users.GetByIdentityRequest\x1a$.account.users.GetByIdentityResponse\"\x00\x42\x30Z.git.perx.ru/perxis/perxis-go/proto/users;usersb\x06proto3')
+
+
+
+_USER = DESCRIPTOR.message_types_by_name['User']
+_FILTER = DESCRIPTOR.message_types_by_name['Filter']
+_CREATEREQUEST = DESCRIPTOR.message_types_by_name['CreateRequest']
+_CREATERESPONSE = DESCRIPTOR.message_types_by_name['CreateResponse']
+_GETREQUEST = DESCRIPTOR.message_types_by_name['GetRequest']
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+_FINDREQUEST = DESCRIPTOR.message_types_by_name['FindRequest']
+_FINDRESPONSE = DESCRIPTOR.message_types_by_name['FindResponse']
+_UPDATEREQUEST = DESCRIPTOR.message_types_by_name['UpdateRequest']
+_DELETEREQUEST = DESCRIPTOR.message_types_by_name['DeleteRequest']
+_GETBYIDENTITYREQUEST = DESCRIPTOR.message_types_by_name['GetByIdentityRequest']
+_GETBYIDENTITYRESPONSE = DESCRIPTOR.message_types_by_name['GetByIdentityResponse']
+User = _reflection.GeneratedProtocolMessageType('User', (_message.Message,), {
+  'DESCRIPTOR' : _USER,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.User)
+  })
+_sym_db.RegisterMessage(User)
+
+Filter = _reflection.GeneratedProtocolMessageType('Filter', (_message.Message,), {
+  'DESCRIPTOR' : _FILTER,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.Filter)
+  })
+_sym_db.RegisterMessage(Filter)
+
+CreateRequest = _reflection.GeneratedProtocolMessageType('CreateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _CREATEREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.CreateRequest)
+  })
+_sym_db.RegisterMessage(CreateRequest)
+
+CreateResponse = _reflection.GeneratedProtocolMessageType('CreateResponse', (_message.Message,), {
+  'DESCRIPTOR' : _CREATERESPONSE,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.CreateResponse)
+  })
+_sym_db.RegisterMessage(CreateResponse)
+
+GetRequest = _reflection.GeneratedProtocolMessageType('GetRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.GetRequest)
+  })
+_sym_db.RegisterMessage(GetRequest)
+
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+FindRequest = _reflection.GeneratedProtocolMessageType('FindRequest', (_message.Message,), {
+  'DESCRIPTOR' : _FINDREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.FindRequest)
+  })
+_sym_db.RegisterMessage(FindRequest)
+
+FindResponse = _reflection.GeneratedProtocolMessageType('FindResponse', (_message.Message,), {
+  'DESCRIPTOR' : _FINDRESPONSE,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.FindResponse)
+  })
+_sym_db.RegisterMessage(FindResponse)
+
+UpdateRequest = _reflection.GeneratedProtocolMessageType('UpdateRequest', (_message.Message,), {
+  'DESCRIPTOR' : _UPDATEREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.UpdateRequest)
+  })
+_sym_db.RegisterMessage(UpdateRequest)
+
+DeleteRequest = _reflection.GeneratedProtocolMessageType('DeleteRequest', (_message.Message,), {
+  'DESCRIPTOR' : _DELETEREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.DeleteRequest)
+  })
+_sym_db.RegisterMessage(DeleteRequest)
+
+GetByIdentityRequest = _reflection.GeneratedProtocolMessageType('GetByIdentityRequest', (_message.Message,), {
+  'DESCRIPTOR' : _GETBYIDENTITYREQUEST,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.GetByIdentityRequest)
+  })
+_sym_db.RegisterMessage(GetByIdentityRequest)
+
+GetByIdentityResponse = _reflection.GeneratedProtocolMessageType('GetByIdentityResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETBYIDENTITYRESPONSE,
+  '__module__' : 'users.users_pb2'
+  # @@protoc_insertion_point(class_scope:account.users.GetByIdentityResponse)
+  })
+_sym_db.RegisterMessage(GetByIdentityResponse)
+
+_USERS = DESCRIPTOR.services_by_name['Users']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z.git.perx.ru/perxis/perxis-go/proto/users;users'
+  _USER._serialized_start=119
+  _USER._serialized_end=324
+  _FILTER._serialized_start=327
+  _FILTER._serialized_end=514
+  _CREATEREQUEST._serialized_start=516
+  _CREATEREQUEST._serialized_end=568
+  _CREATERESPONSE._serialized_start=570
+  _CREATERESPONSE._serialized_end=621
+  _GETREQUEST._serialized_start=623
+  _GETREQUEST._serialized_end=652
+  _GETRESPONSE._serialized_start=654
+  _GETRESPONSE._serialized_end=702
+  _FINDREQUEST._serialized_start=704
+  _FINDREQUEST._serialized_end=794
+  _FINDRESPONSE._serialized_start=796
+  _FINDRESPONSE._serialized_end=861
+  _UPDATEREQUEST._serialized_start=863
+  _UPDATEREQUEST._serialized_end=915
+  _DELETEREQUEST._serialized_start=917
+  _DELETEREQUEST._serialized_end=949
+  _GETBYIDENTITYREQUEST._serialized_start=951
+  _GETBYIDENTITYREQUEST._serialized_end=991
+  _GETBYIDENTITYRESPONSE._serialized_start=993
+  _GETBYIDENTITYRESPONSE._serialized_end=1051
+  _USERS._serialized_start=1054
+  _USERS._serialized_end=1491
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/users/users_pb2_grpc.py b/clients/users/users_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..bad655096accb000f8141615c0b588a9e40c9265
--- /dev/null
+++ b/clients/users/users_pb2_grpc.py
@@ -0,0 +1,242 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from users import users_pb2 as users_dot_users__pb2
+
+
+class UsersStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Create = channel.unary_unary(
+                '/account.users.Users/Create',
+                request_serializer=users_dot_users__pb2.CreateRequest.SerializeToString,
+                response_deserializer=users_dot_users__pb2.CreateResponse.FromString,
+                )
+        self.Get = channel.unary_unary(
+                '/account.users.Users/Get',
+                request_serializer=users_dot_users__pb2.GetRequest.SerializeToString,
+                response_deserializer=users_dot_users__pb2.GetResponse.FromString,
+                )
+        self.Find = channel.unary_unary(
+                '/account.users.Users/Find',
+                request_serializer=users_dot_users__pb2.FindRequest.SerializeToString,
+                response_deserializer=users_dot_users__pb2.FindResponse.FromString,
+                )
+        self.Update = channel.unary_unary(
+                '/account.users.Users/Update',
+                request_serializer=users_dot_users__pb2.UpdateRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.Delete = channel.unary_unary(
+                '/account.users.Users/Delete',
+                request_serializer=users_dot_users__pb2.DeleteRequest.SerializeToString,
+                response_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                )
+        self.GetByIdentity = channel.unary_unary(
+                '/account.users.Users/GetByIdentity',
+                request_serializer=users_dot_users__pb2.GetByIdentityRequest.SerializeToString,
+                response_deserializer=users_dot_users__pb2.GetByIdentityResponse.FromString,
+                )
+
+
+class UsersServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Create(self, request, context):
+        """Создание пользователя или регистрация текущего пользователя в системе, если create.id == `current`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Get(self, request, context):
+        """- Получение любого пользователя по идентификатору
+        - Получение текущего (от лица которого делается запрос) пользователя по специальному идентификатору `current`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Find(self, request, context):
+        """Найти пользователей с фильтрацией. Фильтры доступны пользователям в зависимости от их прав
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Update(self, request, context):
+        """- Обновление существующего пользователя по идентификатору
+        - Обновление текущего пользователя по спец. идентификатору `current`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def Delete(self, request, context):
+        """- Удаление существующего пользователя по идентификатору
+        - Удаление текущего пользователя по спец. идентификатору `current`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+    def GetByIdentity(self, request, context):
+        """Метод для внутреннего использования. Недоступен для внешних
+        запросов - авторизация вернет ошибку `access denied`
+        """
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_UsersServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Create': grpc.unary_unary_rpc_method_handler(
+                    servicer.Create,
+                    request_deserializer=users_dot_users__pb2.CreateRequest.FromString,
+                    response_serializer=users_dot_users__pb2.CreateResponse.SerializeToString,
+            ),
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=users_dot_users__pb2.GetRequest.FromString,
+                    response_serializer=users_dot_users__pb2.GetResponse.SerializeToString,
+            ),
+            'Find': grpc.unary_unary_rpc_method_handler(
+                    servicer.Find,
+                    request_deserializer=users_dot_users__pb2.FindRequest.FromString,
+                    response_serializer=users_dot_users__pb2.FindResponse.SerializeToString,
+            ),
+            'Update': grpc.unary_unary_rpc_method_handler(
+                    servicer.Update,
+                    request_deserializer=users_dot_users__pb2.UpdateRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'Delete': grpc.unary_unary_rpc_method_handler(
+                    servicer.Delete,
+                    request_deserializer=users_dot_users__pb2.DeleteRequest.FromString,
+                    response_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            ),
+            'GetByIdentity': grpc.unary_unary_rpc_method_handler(
+                    servicer.GetByIdentity,
+                    request_deserializer=users_dot_users__pb2.GetByIdentityRequest.FromString,
+                    response_serializer=users_dot_users__pb2.GetByIdentityResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'account.users.Users', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Users(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Create(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/Create',
+            users_dot_users__pb2.CreateRequest.SerializeToString,
+            users_dot_users__pb2.CreateResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/Get',
+            users_dot_users__pb2.GetRequest.SerializeToString,
+            users_dot_users__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Find(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/Find',
+            users_dot_users__pb2.FindRequest.SerializeToString,
+            users_dot_users__pb2.FindResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Update(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/Update',
+            users_dot_users__pb2.UpdateRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def Delete(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/Delete',
+            users_dot_users__pb2.DeleteRequest.SerializeToString,
+            google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
+
+    @staticmethod
+    def GetByIdentity(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.users.Users/GetByIdentity',
+            users_dot_users__pb2.GetByIdentityRequest.SerializeToString,
+            users_dot_users__pb2.GetByIdentityResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/versions/account/versions_pb2.py b/clients/versions/account/versions_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..0924cff3ba2048b18641e5929874fef72f7524b9
--- /dev/null
+++ b/clients/versions/account/versions_pb2.py
@@ -0,0 +1,40 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: versions/account/versions.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1fversions/account/versions.proto\x12\x07\x61\x63\x63ount\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"/\n\x0bGetResponse\x12 \n\x07version\x18\x01 \x01(\x0b\x32\x0f.common.Version2A\n\x08Versions\x12\x35\n\x03Get\x12\x16.google.protobuf.Empty\x1a\x14.account.GetResponse\"\x00\x42=Z;git.perx.ru/perxis/perxis-go/proto/versions/account;accountb\x06proto3')
+
+
+
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'versions.account.versions_pb2'
+  # @@protoc_insertion_point(class_scope:account.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+_VERSIONS = DESCRIPTOR.services_by_name['Versions']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z;git.perx.ru/perxis/perxis-go/proto/versions/account;account'
+  _GETRESPONSE._serialized_start=94
+  _GETRESPONSE._serialized_end=141
+  _VERSIONS._serialized_start=143
+  _VERSIONS._serialized_end=208
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/versions/account/versions_pb2_grpc.py b/clients/versions/account/versions_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..ad4fe2023d2501f33af3b78fd1611ad120a2ff2e
--- /dev/null
+++ b/clients/versions/account/versions_pb2_grpc.py
@@ -0,0 +1,67 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from versions.account import versions_pb2 as versions_dot_account_dot_versions__pb2
+
+
+class VersionsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Get = channel.unary_unary(
+                '/account.Versions/Get',
+                request_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+                response_deserializer=versions_dot_account_dot_versions__pb2.GetResponse.FromString,
+                )
+
+
+class VersionsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_VersionsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                    response_serializer=versions_dot_account_dot_versions__pb2.GetResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'account.Versions', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Versions(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/account.Versions/Get',
+            google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            versions_dot_account_dot_versions__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/clients/versions/content/versions_pb2.py b/clients/versions/content/versions_pb2.py
new file mode 100644
index 0000000000000000000000000000000000000000..12ff3f74909e8055b57149549df17bae69650dff
--- /dev/null
+++ b/clients/versions/content/versions_pb2.py
@@ -0,0 +1,40 @@
+# -*- coding: utf-8 -*-
+# Generated by the protocol buffer compiler.  DO NOT EDIT!
+# source: versions/content/versions.proto
+"""Generated protocol buffer code."""
+from google.protobuf import descriptor as _descriptor
+from google.protobuf import descriptor_pool as _descriptor_pool
+from google.protobuf import message as _message
+from google.protobuf import reflection as _reflection
+from google.protobuf import symbol_database as _symbol_database
+# @@protoc_insertion_point(imports)
+
+_sym_db = _symbol_database.Default()
+
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from common import common_pb2 as common_dot_common__pb2
+
+
+DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\x1fversions/content/versions.proto\x12\x07\x63ontent\x1a\x1bgoogle/protobuf/empty.proto\x1a\x13\x63ommon/common.proto\"/\n\x0bGetResponse\x12 \n\x07version\x18\x01 \x01(\x0b\x32\x0f.common.Version2A\n\x08Versions\x12\x35\n\x03Get\x12\x16.google.protobuf.Empty\x1a\x14.content.GetResponse\"\x00\x42=Z;git.perx.ru/perxis/perxis-go/proto/versions/content;contentb\x06proto3')
+
+
+
+_GETRESPONSE = DESCRIPTOR.message_types_by_name['GetResponse']
+GetResponse = _reflection.GeneratedProtocolMessageType('GetResponse', (_message.Message,), {
+  'DESCRIPTOR' : _GETRESPONSE,
+  '__module__' : 'versions.content.versions_pb2'
+  # @@protoc_insertion_point(class_scope:content.GetResponse)
+  })
+_sym_db.RegisterMessage(GetResponse)
+
+_VERSIONS = DESCRIPTOR.services_by_name['Versions']
+if _descriptor._USE_C_DESCRIPTORS == False:
+
+  DESCRIPTOR._options = None
+  DESCRIPTOR._serialized_options = b'Z;git.perx.ru/perxis/perxis-go/proto/versions/content;content'
+  _GETRESPONSE._serialized_start=94
+  _GETRESPONSE._serialized_end=141
+  _VERSIONS._serialized_start=143
+  _VERSIONS._serialized_end=208
+# @@protoc_insertion_point(module_scope)
diff --git a/clients/versions/content/versions_pb2_grpc.py b/clients/versions/content/versions_pb2_grpc.py
new file mode 100644
index 0000000000000000000000000000000000000000..8cf777985e0d0c74255194e34eebd218a109ec9e
--- /dev/null
+++ b/clients/versions/content/versions_pb2_grpc.py
@@ -0,0 +1,67 @@
+# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
+"""Client and server classes corresponding to protobuf-defined services."""
+import grpc
+
+from google.protobuf import empty_pb2 as google_dot_protobuf_dot_empty__pb2
+from versions.content import versions_pb2 as versions_dot_content_dot_versions__pb2
+
+
+class VersionsStub(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def __init__(self, channel):
+        """Constructor.
+
+        Args:
+            channel: A grpc.Channel.
+        """
+        self.Get = channel.unary_unary(
+                '/content.Versions/Get',
+                request_serializer=google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+                response_deserializer=versions_dot_content_dot_versions__pb2.GetResponse.FromString,
+                )
+
+
+class VersionsServicer(object):
+    """Missing associated documentation comment in .proto file."""
+
+    def Get(self, request, context):
+        """Missing associated documentation comment in .proto file."""
+        context.set_code(grpc.StatusCode.UNIMPLEMENTED)
+        context.set_details('Method not implemented!')
+        raise NotImplementedError('Method not implemented!')
+
+
+def add_VersionsServicer_to_server(servicer, server):
+    rpc_method_handlers = {
+            'Get': grpc.unary_unary_rpc_method_handler(
+                    servicer.Get,
+                    request_deserializer=google_dot_protobuf_dot_empty__pb2.Empty.FromString,
+                    response_serializer=versions_dot_content_dot_versions__pb2.GetResponse.SerializeToString,
+            ),
+    }
+    generic_handler = grpc.method_handlers_generic_handler(
+            'content.Versions', rpc_method_handlers)
+    server.add_generic_rpc_handlers((generic_handler,))
+
+
+ # This class is part of an EXPERIMENTAL API.
+class Versions(object):
+    """Missing associated documentation comment in .proto file."""
+
+    @staticmethod
+    def Get(request,
+            target,
+            options=(),
+            channel_credentials=None,
+            call_credentials=None,
+            insecure=False,
+            compression=None,
+            wait_for_ready=None,
+            timeout=None,
+            metadata=None):
+        return grpc.experimental.unary_unary(request, target, '/content.Versions/Get',
+            google_dot_protobuf_dot_empty__pb2.Empty.SerializeToString,
+            versions_dot_content_dot_versions__pb2.GetResponse.FromString,
+            options, channel_credentials,
+            insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
diff --git a/perxis-proto b/perxis-proto
new file mode 160000
index 0000000000000000000000000000000000000000..ed11d439846b40a3e78e1fe8483569e32653ac22
--- /dev/null
+++ b/perxis-proto
@@ -0,0 +1 @@
+Subproject commit ed11d439846b40a3e78e1fe8483569e32653ac22