mirror of
https://gitlab.com/bramw/baserow.git
synced 2025-04-04 21:25:24 +00:00
Resolve "Every snapshot causes additional queries to check_permissions when rbac is enabled"
This commit is contained in:
parent
a4eed09b2d
commit
5d195c1523
28 changed files with 367 additions and 159 deletions
backend
src/baserow
api/snapshots
config/settings
contrib
core
tests/baserow
changelog/entries/unreleased/bug
docs/installation
enterprise/backend
src/baserow_enterprise
tests/baserow_enterprise_tests/role
|
@ -135,7 +135,9 @@ class SnapshotsView(APIView):
|
|||
"""
|
||||
|
||||
handler = SnapshotHandler()
|
||||
snapshot_created = handler.create(application_id, request.user, data["name"])
|
||||
snapshot_created = handler.start_create_job(
|
||||
application_id, request.user, data["name"]
|
||||
)
|
||||
serializer = JobSerializer(snapshot_created["job"])
|
||||
return Response(serializer.data, status=HTTP_202_ACCEPTED)
|
||||
|
||||
|
@ -195,7 +197,7 @@ class RestoreSnapshotView(APIView):
|
|||
"""
|
||||
|
||||
handler = SnapshotHandler()
|
||||
job = handler.restore(snapshot_id, request.user)
|
||||
job = handler.start_restore_job(snapshot_id, request.user)
|
||||
serializer = JobSerializer(job)
|
||||
return Response(serializer.data)
|
||||
|
||||
|
|
|
@ -1075,7 +1075,7 @@ BASEROW_ROW_HISTORY_RETENTION_DAYS = int(
|
|||
BASEROW_MAX_ROW_REPORT_ERROR_COUNT = int(
|
||||
os.getenv("BASEROW_MAX_ROW_REPORT_ERROR_COUNT", 30)
|
||||
)
|
||||
BASEROW_MAX_SNAPSHOTS_PER_GROUP = int(os.getenv("BASEROW_MAX_SNAPSHOTS_PER_GROUP", -1))
|
||||
BASEROW_MAX_SNAPSHOTS_PER_GROUP = int(os.getenv("BASEROW_MAX_SNAPSHOTS_PER_GROUP", 50))
|
||||
BASEROW_SNAPSHOT_EXPIRATION_TIME_DAYS = int(
|
||||
os.getenv("BASEROW_SNAPSHOT_EXPIRATION_TIME_DAYS", 360) # 360 days
|
||||
)
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.data_sources.models import DataSource
|
||||
from baserow.contrib.builder.object_scopes import BuilderObjectScopeType
|
||||
|
@ -19,9 +19,16 @@ class BuilderDataSourceObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("builder_page")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"page", "page__builder", "page__builder__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(page__builder__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"page__builder__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.domains.models import Domain
|
||||
from baserow.contrib.builder.object_scopes import BuilderObjectScopeType
|
||||
|
@ -22,9 +22,16 @@ class BuilderDomainObjectScopeType(ObjectScopeType):
|
|||
def get_parent(self, context: ContextObject) -> Optional[ContextObject]:
|
||||
return context.builder
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"builder", "builder__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(builder__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"builder__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.elements.models import Element
|
||||
from baserow.contrib.builder.object_scopes import BuilderObjectScopeType
|
||||
|
@ -19,9 +19,16 @@ class BuilderElementObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("builder_page")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"page", "page__builder", "page__builder__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(page__builder__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"page__builder__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.models import Builder
|
||||
from baserow.core.object_scopes import (
|
||||
|
@ -15,8 +15,11 @@ class BuilderObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("application")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return super().get_base_queryset(include_trash)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.object_scopes import BuilderObjectScopeType
|
||||
from baserow.contrib.builder.pages.models import Page
|
||||
|
@ -18,9 +18,16 @@ class BuilderPageObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("builder")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"builder", "builder__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(builder__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"builder__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.builder.elements.object_scopes import BuilderElementObjectScopeType
|
||||
from baserow.contrib.builder.object_scopes import BuilderObjectScopeType
|
||||
|
@ -18,7 +18,19 @@ class BuilderWorkflowActionScopeType(ObjectScopeType):
|
|||
model_class = BuilderWorkflowAction
|
||||
|
||||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("builder_element")
|
||||
return object_scope_type_registry.get_by_type(BuilderElementObjectScopeType)
|
||||
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(page__builder__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"page__builder__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.database.models import Field
|
||||
from baserow.contrib.database.object_scopes import DatabaseObjectScopeType
|
||||
|
@ -17,9 +17,18 @@ class FieldObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_table")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"table", "table__database", "table__database__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(
|
||||
table__database__workspace__isnull=False,
|
||||
)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False):
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -67,4 +67,8 @@ __all__ = [
|
|||
|
||||
class Database(Application):
|
||||
def get_parent(self):
|
||||
# This is a bit of a hack to prevent an unecesary query to the database to
|
||||
# get the parent workspace that we already have.
|
||||
if "workspace" in self._state.fields_cache:
|
||||
self.application_ptr.workspace = self.workspace
|
||||
return self.application_ptr
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.database.models import Database
|
||||
from baserow.core.object_scopes import (
|
||||
|
@ -15,8 +15,11 @@ class DatabaseObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("application")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet[Database]:
|
||||
return super().get_base_queryset(include_trash).filter(workspace__isnull=False)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet[Database]:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.database.object_scopes import DatabaseObjectScopeType
|
||||
from baserow.contrib.database.table.models import Table
|
||||
|
@ -16,9 +16,16 @@ class DatabaseTableObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"database", "database__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.contrib.database.object_scopes import DatabaseObjectScopeType
|
||||
from baserow.contrib.database.table.object_scopes import DatabaseTableObjectScopeType
|
||||
|
@ -24,9 +24,16 @@ class DatabaseViewObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_table")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"table", "table__database", "table__database__workspace"
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
@ -52,12 +59,16 @@ class DatabaseViewDecorationObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_view")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"view",
|
||||
"view__table",
|
||||
"view__table__database",
|
||||
"view__table__database__workspace",
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(view__table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"view__table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
@ -86,12 +97,16 @@ class DatabaseViewSortObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_view")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"view",
|
||||
"view__table",
|
||||
"view__table__database",
|
||||
"view__table__database__workspace",
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(view__table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"view__table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
@ -120,12 +135,16 @@ class DatabaseViewFilterObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_view")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"view",
|
||||
"view__table",
|
||||
"view__table__database",
|
||||
"view__table__database__workspace",
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(view__table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"view__table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
@ -154,12 +173,16 @@ class DatabaseViewFilterGroupObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_view")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"view",
|
||||
"view__table",
|
||||
"view__table__database",
|
||||
"view__table__database__workspace",
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(view__table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"view__table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
@ -188,12 +211,16 @@ class DatabaseViewGroupByObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("database_view")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"view",
|
||||
"view__table",
|
||||
"view__table__database",
|
||||
"view__table__database__workspace",
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return (
|
||||
super()
|
||||
.get_base_queryset(include_trash)
|
||||
.filter(view__table__database__workspace__isnull=False)
|
||||
)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"view__table__database__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.core.integrations.models import Integration
|
||||
from baserow.core.object_scopes import (
|
||||
|
@ -17,9 +17,9 @@ class IntegrationObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("application")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"application", "application__workspace"
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"application__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -362,7 +362,8 @@ class HierarchicalModelMixin(models.Model, metaclass=AbstractModelMeta):
|
|||
This mixin introduce some helpers for working with hierarchical models.
|
||||
"""
|
||||
|
||||
@abc.abstractclassmethod
|
||||
@classmethod
|
||||
@abc.abstractmethod
|
||||
def get_parent(self):
|
||||
"""
|
||||
:return: The parent of this model. Returns None if this is the root.
|
||||
|
|
|
@ -419,13 +419,11 @@ class Application(
|
|||
return cls.get_highest_order_of_queryset(queryset) + 1
|
||||
|
||||
def get_parent(self):
|
||||
# If this application is an application snapshot, then it'll
|
||||
# have a None workspace, so instead we define its parent as
|
||||
# the source snapshot's `snapshot_from`.
|
||||
if self.workspace_id:
|
||||
return self.workspace
|
||||
else:
|
||||
return self.snapshot_from.get()
|
||||
if not self.workspace_id:
|
||||
raise ValueError(
|
||||
"Cannot call get_parent if workspace is None. Please check your hierarchy."
|
||||
)
|
||||
return self.workspace
|
||||
|
||||
|
||||
class TemplateCategory(models.Model):
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.core.models import (
|
||||
Application,
|
||||
|
@ -32,8 +32,11 @@ class ApplicationObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("workspace")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return super().get_base_queryset(include_trash).filter(workspace__isnull=False)
|
||||
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
@ -49,8 +52,8 @@ class WorkspaceInvitationObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("workspace")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
@ -66,8 +69,8 @@ class WorkspaceUserObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("workspace")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
|
|
@ -769,20 +769,29 @@ class ObjectScopeType(Instance, ModelInstanceMixin):
|
|||
f"Must be implemented by the specific type <{self.type}>"
|
||||
)
|
||||
|
||||
def get_base_queryset(self) -> QuerySet:
|
||||
def get_base_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
"""
|
||||
:params include_trash: If true then also includes trashed objects in the
|
||||
queryset. Needed to verify if a user needs to be included in the recipient
|
||||
list of a deleted_* signal.
|
||||
Returns the base queryset for the objects of this scope
|
||||
"""
|
||||
|
||||
return self.model_class.objects.all()
|
||||
model_manager = self.model_class.objects
|
||||
if include_trash and hasattr(self.model_class, "objects_and_trash"):
|
||||
model_manager = self.model_class.objects_and_trash
|
||||
return model_manager.order_by().all()
|
||||
|
||||
def get_enhanced_queryset(self) -> QuerySet:
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
"""
|
||||
:params include_trash: If true then also includes trashed objects in the
|
||||
queryset. Needed to verify if a user needs to be included in the recipient
|
||||
list of a deleted_* signal.
|
||||
Returns the enhanced queryset for the objects of this scope enhanced for better
|
||||
performances.
|
||||
"""
|
||||
|
||||
return self.get_base_queryset()
|
||||
return self.get_base_queryset(include_trash=include_trash)
|
||||
|
||||
def are_objects_child_of(
|
||||
self, child_objects: List[Any], parent_object: ScopeObject
|
||||
|
@ -849,14 +858,18 @@ class ObjectScopeType(Instance, ModelInstanceMixin):
|
|||
"""
|
||||
|
||||
objects_per_scope = {}
|
||||
parent_scope_types = set()
|
||||
|
||||
parent_scopes = []
|
||||
for scope in scopes:
|
||||
if object_scope_type_registry.get_by_model(scope).type == self.type:
|
||||
object_scope_type = object_scope_type_registry.get_by_model(scope)
|
||||
if object_scope_type.type == self.type:
|
||||
# Scope of the same type doesn't need to be queried
|
||||
objects_per_scope[scope] = set([scope])
|
||||
else:
|
||||
parent_scopes.append(scope)
|
||||
objects_per_scope[scope] = set()
|
||||
parent_scope_types.add(object_scope_type)
|
||||
|
||||
if parent_scopes:
|
||||
query_result = list(
|
||||
|
@ -867,15 +880,13 @@ class ObjectScopeType(Instance, ModelInstanceMixin):
|
|||
|
||||
# We have all the objects in the queryset, but now we want to sort them
|
||||
# into buckets per original scope they are a child of.
|
||||
for scope in parent_scopes:
|
||||
objects_per_scope[scope] = set()
|
||||
scope_type = object_scope_type_registry.get_by_model(scope)
|
||||
for obj in query_result:
|
||||
for obj in query_result:
|
||||
for scope_type in parent_scope_types:
|
||||
parent_scope = object_scope_type_registry.get_parent(
|
||||
obj, at_scope_type=scope_type
|
||||
)
|
||||
if parent_scope == scope:
|
||||
objects_per_scope[scope].add(obj)
|
||||
if parent_scope in objects_per_scope:
|
||||
objects_per_scope[parent_scope].add(obj)
|
||||
|
||||
return objects_per_scope
|
||||
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
import datetime
|
||||
|
||||
from django.conf import settings
|
||||
from django.contrib.auth.models import AbstractUser
|
||||
from django.core.files.storage import default_storage
|
||||
from django.db import IntegrityError, OperationalError
|
||||
from django.db.models.query import QuerySet
|
||||
|
@ -17,7 +18,7 @@ from baserow.core.exceptions import (
|
|||
from baserow.core.handler import CoreHandler
|
||||
from baserow.core.jobs.handler import JobHandler
|
||||
from baserow.core.jobs.models import Job
|
||||
from baserow.core.models import Application, Snapshot, User, Workspace
|
||||
from baserow.core.models import Application, Snapshot, Workspace
|
||||
from baserow.core.registries import ImportExportConfig, application_type_registry
|
||||
from baserow.core.signals import application_created
|
||||
from baserow.core.snapshots.exceptions import (
|
||||
|
@ -77,7 +78,7 @@ class SnapshotHandler:
|
|||
if snapshot.mark_for_deletion is True:
|
||||
raise SnapshotIsBeingDeleted()
|
||||
|
||||
def list(self, application_id: int, performed_by: User) -> QuerySet:
|
||||
def list(self, application_id: int, performed_by: AbstractUser) -> QuerySet:
|
||||
"""
|
||||
Lists all snapshots for the given application id if the provided
|
||||
user is in the same workspace as the application.
|
||||
|
@ -122,31 +123,46 @@ class SnapshotHandler:
|
|||
.order_by("-created_at", "-id")
|
||||
)
|
||||
|
||||
def create(self, application_id: int, performed_by: User, name: str):
|
||||
def start_create_job(
|
||||
self, application_id: int, performed_by: AbstractUser, name: str
|
||||
):
|
||||
"""
|
||||
Creates a new application snapshot of the given application if the provided
|
||||
user is in the same workspace as the application.
|
||||
Create a snapshot instance to track the creation of a snapshot and start the job
|
||||
to perform the snapshot creation.
|
||||
|
||||
:param application_id: The ID of the application for which to list
|
||||
snapshots.
|
||||
:param performed_by: The user performing the operation that should
|
||||
have sufficient permissions.
|
||||
:param application_id: The ID of the application for which to list snapshots.
|
||||
:param performed_by: The user performing the operation that should have
|
||||
sufficient permissions.
|
||||
:param name: The name for the new snapshot.
|
||||
:raises ApplicationDoesNotExist: When the application with the provided id
|
||||
does not exist.
|
||||
:raises ApplicationDoesNotExist: When the application with the provided id does
|
||||
not exist.
|
||||
:raises UserNotInWorkspace: When the user doesn't belong to the same workspace
|
||||
as the application.
|
||||
:raises MaximumSnapshotsReached: When the workspace has already reached
|
||||
the maximum of allowed snapshots.
|
||||
:raises ApplicationOperationNotSupported: When the application type
|
||||
doesn't support creating snapshots.
|
||||
:raises SnapshotIsBeingCreated: When creating a snapshot is already
|
||||
scheduled for the application.
|
||||
:raises MaxJobCountExceeded: When the user already has a running
|
||||
job to create a snapshot of the same type.
|
||||
:return: The snapshot object that was created.
|
||||
:raises MaximumSnapshotsReached: When the workspace has already reached the
|
||||
maximum of allowed snapshots.
|
||||
:raises ApplicationOperationNotSupported: When the application type doesn't
|
||||
support creating snapshots.
|
||||
:raises SnapshotIsBeingCreated: When creating a snapshot is already scheduled
|
||||
for the application.
|
||||
:raises MaxJobCountExceeded: When the user already has a running job to create a
|
||||
snapshot of the same type.
|
||||
:return: The snapshot object that was created and the started job.
|
||||
"""
|
||||
|
||||
snapshot = self.create(application_id, performed_by, name)
|
||||
|
||||
job = JobHandler().create_and_start_job(
|
||||
performed_by,
|
||||
CreateSnapshotJobType.type,
|
||||
snapshot=snapshot,
|
||||
)
|
||||
|
||||
return {
|
||||
"snapshot": snapshot,
|
||||
"job": job,
|
||||
}
|
||||
|
||||
def create(self, application_id, performed_by, name):
|
||||
try:
|
||||
application = (
|
||||
Application.objects.filter(id=application_id)
|
||||
|
@ -192,23 +208,12 @@ class SnapshotHandler:
|
|||
if "unique constraint" in e.args[0]:
|
||||
raise SnapshotNameNotUnique()
|
||||
raise e
|
||||
return snapshot
|
||||
|
||||
job = JobHandler().create_and_start_job(
|
||||
performed_by,
|
||||
CreateSnapshotJobType.type,
|
||||
False,
|
||||
snapshot=snapshot,
|
||||
)
|
||||
|
||||
return {
|
||||
"snapshot": snapshot,
|
||||
"job": job,
|
||||
}
|
||||
|
||||
def restore(
|
||||
def start_restore_job(
|
||||
self,
|
||||
snapshot_id: int,
|
||||
performed_by: User,
|
||||
performed_by: AbstractUser,
|
||||
) -> Job:
|
||||
"""
|
||||
Restores a previously created snapshot with the given ID if the
|
||||
|
@ -262,7 +267,6 @@ class SnapshotHandler:
|
|||
job = JobHandler().create_and_start_job(
|
||||
performed_by,
|
||||
RestoreSnapshotJobType.type,
|
||||
False,
|
||||
snapshot=snapshot,
|
||||
)
|
||||
|
||||
|
@ -274,7 +278,7 @@ class SnapshotHandler:
|
|||
if snapshot.snapshot_to_application is not None:
|
||||
delete_application_snapshot.delay(snapshot.snapshot_to_application.id)
|
||||
|
||||
def delete(self, snapshot_id: int, performed_by: User) -> None:
|
||||
def delete(self, snapshot_id: int, performed_by: AbstractUser) -> None:
|
||||
"""
|
||||
Deletes a previously created snapshot with the given ID if the
|
||||
provided user belongs to the same workspace as the application.
|
||||
|
@ -444,6 +448,9 @@ class SnapshotHandler:
|
|||
restore_snapshot_import_export_config = ImportExportConfig(
|
||||
include_permission_data=True, reduce_disk_space_usage=False
|
||||
)
|
||||
# Temporary set the workspace for the application so that the permissions can
|
||||
# be correctly set during the import process.
|
||||
application.workspace = workspace
|
||||
exported_application = application_type.export_serialized(
|
||||
application, restore_snapshot_import_export_config, None, default_storage
|
||||
)
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.core.models import Snapshot
|
||||
from baserow.core.object_scopes import (
|
||||
|
@ -15,9 +15,9 @@ class SnapshotObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("application")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"snapshot_from_application", "snapshot_from_application__workspace"
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"snapshot_from_application__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
from typing import Optional
|
||||
|
||||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.core.object_scopes import (
|
||||
ApplicationObjectScopeType,
|
||||
|
@ -17,9 +17,9 @@ class UserSourceObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self) -> Optional["ObjectScopeType"]:
|
||||
return object_scope_type_registry.get("application")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related(
|
||||
"application", "application__workspace"
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related(
|
||||
"application__workspace"
|
||||
)
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
|
|
|
@ -11,9 +11,9 @@ def test_can_create_a_snapshot_for_builder_application(data_fixture):
|
|||
user = data_fixture.create_user()
|
||||
builder = data_fixture.create_builder_application(user)
|
||||
|
||||
snapshot = SnapshotHandler().create(builder.id, performed_by=user, name="test")[
|
||||
"snapshot"
|
||||
]
|
||||
snapshot = SnapshotHandler().start_create_job(
|
||||
builder.id, performed_by=user, name="test"
|
||||
)["snapshot"]
|
||||
|
||||
assert snapshot is not None
|
||||
assert snapshot.name == "test"
|
||||
|
@ -25,9 +25,9 @@ def test_can_delete_a_snapshot_for_builder_application(data_fixture):
|
|||
user = data_fixture.create_user()
|
||||
builder = data_fixture.create_builder_application(user)
|
||||
|
||||
snapshot = SnapshotHandler().create(builder.id, performed_by=user, name="test")[
|
||||
"snapshot"
|
||||
]
|
||||
snapshot = SnapshotHandler().start_create_job(
|
||||
builder.id, performed_by=user, name="test"
|
||||
)["snapshot"]
|
||||
|
||||
SnapshotHandler().delete(snapshot.id, user)
|
||||
|
||||
|
|
|
@ -29,7 +29,9 @@ def test_create_snapshot_action_type(
|
|||
application = data_fixture.create_database_application(workspace=workspace)
|
||||
|
||||
with django_capture_on_commit_callbacks(execute=True):
|
||||
result = SnapshotHandler().create(application.id, user, "test snapshot")
|
||||
result = SnapshotHandler().start_create_job(
|
||||
application.id, user, "test snapshot"
|
||||
)
|
||||
|
||||
snapshot = result["snapshot"]
|
||||
job = result["job"]
|
||||
|
@ -68,7 +70,7 @@ def test_restore_snapshot_action_type(
|
|||
)
|
||||
|
||||
with django_capture_on_commit_callbacks(execute=True):
|
||||
job = SnapshotHandler().restore(snapshot.id, user)
|
||||
job = SnapshotHandler().start_restore_job(snapshot.id, user)
|
||||
|
||||
job.refresh_from_db()
|
||||
assert job.state == JOB_FINISHED
|
||||
|
|
|
@ -0,0 +1,7 @@
|
|||
{
|
||||
"type": "bug",
|
||||
"message": "Fix a bug causing every snapshot to perform additional queries to check permissions for RBAC.",
|
||||
"issue_number": 2941,
|
||||
"bullet_points": [],
|
||||
"created_at": "2024-08-27"
|
||||
}
|
|
@ -273,7 +273,7 @@ domain than your Baserow, you need to make sure CORS is configured correctly.
|
|||
| BASEROW\_DISABLE\_PUBLIC\_URL\_CHECK | When opening the Baserow login page a check is run to ensure the PUBLIC\_BACKEND\_URL/BASEROW\_PUBLIC\_URL variables are set correctly and your browser can correctly connect to the backend. If misconfigured an error is shown. If you wish to disable this check and warning set this to any non empty value. | |
|
||||
| ADDITIONAL\_MODULES | **Internal** A list of file paths to Nuxt module.js files to load as additional Nuxt modules into Baserow on startup. | |
|
||||
| BASEROW\_DISABLE\_GOOGLE\_DOCS\_FILE\_PREVIEW | Set to \`true\` or \`1\` to disable Google docs file preview. | |
|
||||
| BASEROW_MAX_SNAPSHOTS_PER_GROUP | Controls how many application snapshots can be created per workspace. | -1 (unlimited) |
|
||||
| BASEROW_MAX_SNAPSHOTS_PER_GROUP | Controls how many application snapshots can be created per workspace. | 50 |
|
||||
| BASEROW\_USE\_PG\_FULLTEXT\_SEARCH | By default, Baserow will use Postgres full-text as its search backend. If the product is installed on a system with limited disk space, and less accurate results / degraded search performance is acceptable, then switch this setting off by setting it to false. | true |
|
||||
| BASEROW\_UNIQUE\_ROW\_VALUES\_SIZE\_LIMIT | Sets the limit for the automatic detection of multiselect options when converting a text field to a multiselect field. Increase the value to detect more options automatically, but consider performance implications. | 100 |
|
||||
| BASEROW\_BUILDER\_DOMAINS | A comma separated list of domain names that can be used as the domains to create sub domains in the application builder. | |
|
||||
|
|
|
@ -477,6 +477,12 @@ class RoleAssignmentHandler:
|
|||
roles_per_scope_per_user = defaultdict(list)
|
||||
for actor in actors:
|
||||
for key, value in roles_by_scope[actor.id].items():
|
||||
# scope_cache contains all the filtered scopes we need to check
|
||||
# permissions for. Objects from snapshotted applications (table,
|
||||
# applications, etc.) are filtered out as they're not accessible to the
|
||||
# user, so we can safely ignore them here.
|
||||
if key not in scope_cache:
|
||||
continue
|
||||
roles_per_scope_per_user[actor].append((scope_cache[key], value))
|
||||
|
||||
return roles_per_scope_per_user
|
||||
|
@ -744,17 +750,23 @@ class RoleAssignmentHandler:
|
|||
|
||||
return self.assign_role_batch(workspace, new_role_assignments)
|
||||
|
||||
def get_scopes(self, content_type_id, scope_ids) -> List[ScopeObject]:
|
||||
def get_scopes(self, content_type_id, scope_ids) -> QuerySet[ScopeObject]:
|
||||
"""
|
||||
Helper method that returns the actual scope object given the scope ID and
|
||||
the scope type.
|
||||
|
||||
:param scope_id: The scope id.
|
||||
:param content_type_id: The content_type id
|
||||
:return: A QuerySet of the scope objects matching the given scope IDs.
|
||||
"""
|
||||
|
||||
content_type = ContentType.objects.get_for_id(content_type_id)
|
||||
return content_type.get_all_objects_for_this_type(id__in=scope_ids)
|
||||
object_scope = object_scope_type_registry.get_for_class(
|
||||
content_type.model_class()
|
||||
)
|
||||
return object_scope.get_enhanced_queryset(include_trash=True).filter(
|
||||
id__in=scope_ids
|
||||
)
|
||||
|
||||
def get_role_assignments(self, workspace: Workspace, scope: Optional[ScopeObject]):
|
||||
"""
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
from django.db.models import Q
|
||||
from django.db.models import Q, QuerySet
|
||||
|
||||
from baserow.core.object_scopes import WorkspaceObjectScopeType
|
||||
from baserow.core.registries import ObjectScopeType, object_scope_type_registry
|
||||
|
@ -12,8 +12,8 @@ class TeamObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("workspace")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("workspace")
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
@ -29,8 +29,8 @@ class TeamSubjectObjectScopeType(ObjectScopeType):
|
|||
def get_parent_scope(self):
|
||||
return object_scope_type_registry.get("team")
|
||||
|
||||
def get_enhanced_queryset(self):
|
||||
return self.get_base_queryset().prefetch_related("team", "team__workspace")
|
||||
def get_enhanced_queryset(self, include_trash: bool = False) -> QuerySet:
|
||||
return self.get_base_queryset(include_trash).select_related("team__workspace")
|
||||
|
||||
def get_filter_for_scope_type(self, scope_type, scopes):
|
||||
if scope_type.type == WorkspaceObjectScopeType.type:
|
||||
|
|
|
@ -28,6 +28,7 @@ from baserow.contrib.database.table.operations import (
|
|||
UpdateDatabaseTableOperationType,
|
||||
)
|
||||
from baserow.core.exceptions import PermissionException
|
||||
from baserow.core.handler import CoreHandler
|
||||
from baserow.core.models import Application
|
||||
from baserow.core.notifications.operations import (
|
||||
ClearNotificationsOperationType,
|
||||
|
@ -47,7 +48,9 @@ from baserow.core.operations import (
|
|||
UpdateWorkspaceOperationType,
|
||||
)
|
||||
from baserow.core.registries import operation_type_registry
|
||||
from baserow.core.snapshots.handler import SnapshotHandler
|
||||
from baserow.core.types import PermissionCheck
|
||||
from baserow.core.utils import Progress
|
||||
from baserow_enterprise.role.default_roles import default_roles
|
||||
from baserow_enterprise.role.handler import RoleAssignmentHandler
|
||||
from baserow_enterprise.role.models import Role
|
||||
|
@ -1775,3 +1778,72 @@ def test_check_multiple_permissions_perf(
|
|||
for q in captured.captured_queries:
|
||||
print(q)
|
||||
print(len(captured.captured_queries))
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_fetching_permissions_does_not_extra_queries_per_snapshot(
|
||||
data_fixture, enterprise_data_fixture, synced_roles
|
||||
):
|
||||
enterprise_data_fixture.enable_enterprise()
|
||||
admin = data_fixture.create_user()
|
||||
viewer = data_fixture.create_user()
|
||||
|
||||
workspace = data_fixture.create_workspace(
|
||||
members=[admin, viewer],
|
||||
)
|
||||
database = data_fixture.create_database_application(workspace=workspace, order=1)
|
||||
table = data_fixture.create_database_table(user=admin, database=database)
|
||||
|
||||
role_admin = Role.objects.get(uid="ADMIN")
|
||||
role_viewer = Role.objects.get(uid="VIEWER")
|
||||
RoleAssignmentHandler().assign_role(admin, workspace, role=role_admin)
|
||||
RoleAssignmentHandler().assign_role(viewer, workspace, role=role_viewer)
|
||||
RoleAssignmentHandler().assign_role(
|
||||
admin, workspace, role=role_admin, scope=database.application_ptr
|
||||
)
|
||||
RoleAssignmentHandler().assign_role(
|
||||
viewer, workspace, role=role_viewer, scope=database.application_ptr
|
||||
)
|
||||
|
||||
# The first time it also fetches the settings and the content types
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
with CaptureQueriesContext(connection) as captured_1:
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
# Let's create a snapshot of the database
|
||||
handler = SnapshotHandler()
|
||||
snapshot = handler.create(database.id, admin, "Test snapshot")
|
||||
handler.perform_create(snapshot, Progress(100))
|
||||
|
||||
with CaptureQueriesContext(connection) as captured_2:
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
assert len(captured_2.captured_queries) == len(captured_1.captured_queries)
|
||||
|
||||
# Another snapshot won't increase the number of queries
|
||||
snapshot = handler.create(database.id, admin, "Test snapshot 2")
|
||||
handler.perform_create(snapshot, Progress(100))
|
||||
|
||||
with CaptureQueriesContext(connection) as captured_3:
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
assert len(captured_3.captured_queries) == len(captured_2.captured_queries)
|
||||
|
||||
# The same should be valid for builder applications
|
||||
|
||||
builder = data_fixture.create_builder_application(user=admin, workspace=workspace)
|
||||
page = data_fixture.create_builder_page(builder=builder)
|
||||
|
||||
with CaptureQueriesContext(connection) as captured_1:
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
# Let's create a snapshot of the builder app
|
||||
handler = SnapshotHandler()
|
||||
snapshot = handler.create(builder.id, admin, "Test snapshot")
|
||||
handler.perform_create(snapshot, Progress(100))
|
||||
|
||||
with CaptureQueriesContext(connection) as captured_2:
|
||||
CoreHandler().get_permissions(viewer, workspace=workspace)
|
||||
|
||||
assert len(captured_1.captured_queries) == len(captured_2.captured_queries)
|
||||
|
|
Loading…
Add table
Reference in a new issue