Make sure all log events have portfolio and app ids (where applicable)

This commit is contained in:
leigh-mil 2019-05-16 16:58:41 -04:00
parent 927d1b7925
commit a1eb7ec935
8 changed files with 73 additions and 36 deletions

View File

@ -14,7 +14,9 @@ class ApplicationRoles(object):
@classmethod @classmethod
def create(cls, user, application, permission_set_names): def create(cls, user, application, permission_set_names):
application_role = ApplicationRole(user=user, application_id=application.id) application_role = ApplicationRole(
user=user, application_id=application.id, application=application
)
application_role.permission_sets = ApplicationRoles._permission_sets_for_names( application_role.permission_sets = ApplicationRoles._permission_sets_for_names(
permission_set_names permission_set_names

View File

@ -1,5 +1,3 @@
from sqlalchemy import or_
from atst.database import db from atst.database import db
from atst.domain.common import Query from atst.domain.common import Query
from atst.models.audit_event import AuditEvent from atst.models.audit_event import AuditEvent

View File

@ -26,6 +26,10 @@ class ApplicationInvitation(Base, TimestampsMixin, AuditableMixin, InvitesMixin)
def application_id(self): def application_id(self):
return self.role.application_id return self.role.application_id
@property
def portfolio_id(self):
return self.role.portfolio_id
@property @property
def event_details(self): def event_details(self):
return {"email": self.email, "dod_id": self.user_dod_id} return {"email": self.email, "dod_id": self.user_dod_id}

View File

@ -75,6 +75,9 @@ class ApplicationRole(
lambda prms: prms.name == perm_set_name, self.permission_sets lambda prms: prms.name == perm_set_name, self.permission_sets
) )
@property
def portfolio_id(self):
return self.application.portfolio_id
Index( Index(
"application_role_user_application", "application_role_user_application",

View File

@ -35,7 +35,8 @@ class Environment(
def portfolio(self): def portfolio(self):
return self.application.portfolio return self.application.portfolio
def auditable_portfolio_id(self): @property
def portfolio_id(self):
return self.application.portfolio_id return self.application.portfolio_id
def __repr__(self): def __repr__(self):

View File

@ -38,6 +38,14 @@ class EnvironmentRole(
def history(self): def history(self):
return self.get_changes() return self.get_changes()
@property
def portfolio_id(self):
return self.environment.application.portfolio_id
@property
def application_id(self):
return self.environment.application_id
@property @property
def displayname(self): def displayname(self):
return self.role return self.role

View File

@ -13,33 +13,20 @@ class AuditableMixin(object):
@staticmethod @staticmethod
def create_audit_event(connection, resource, action, changed_state=None): def create_audit_event(connection, resource, action, changed_state=None):
user_id = getattr_path(g, "current_user.id") user_id = getattr_path(g, "current_user.id")
resource_type = resource.resource_type
display_name = resource.displayname
event_details = resource.event_details
if resource_type == "portfolio":
portfolio_id = resource.id
else:
portfolio_id = resource.portfolio_id
if resource_type == "application":
application_id = resource.id
else:
application_id = resource.application_id
if changed_state is None: if changed_state is None:
changed_state = resource.history if action == ACTION_UPDATE else None changed_state = resource.history if action == ACTION_UPDATE else None
audit_event = AuditEvent( audit_event = AuditEvent(
user_id=user_id, user_id=user_id,
portfolio_id=portfolio_id, portfolio_id=resource.portfolio_id,
application_id=application_id, application_id=resource.application_id,
resource_type=resource_type, resource_type=resource.resource_type,
resource_id=resource.id, resource_id=resource.id,
display_name=display_name, display_name=resource.displayname,
action=action, action=action,
changed_state=changed_state, changed_state=changed_state,
event_details=event_details, event_details=resource.event_details,
) )
app.logger.info( app.logger.info(

View File

@ -4,12 +4,17 @@ from atst.domain.applications import Applications
from atst.domain.audit_log import AuditLog from atst.domain.audit_log import AuditLog
from atst.domain.exceptions import UnauthorizedError from atst.domain.exceptions import UnauthorizedError
from atst.domain.permission_sets import PermissionSets from atst.domain.permission_sets import PermissionSets
from atst.domain.portfolios import Portfolios
from atst.models.portfolio_role import Status as PortfolioRoleStatus from atst.models.portfolio_role import Status as PortfolioRoleStatus
from tests.factories import ( from tests.factories import (
UserFactory, ApplicationFactory,
ApplicationInvitationFactory,
ApplicationRoleFactory,
EnvironmentFactory,
EnvironmentRoleFactory,
PortfolioFactory, PortfolioFactory,
PortfolioRoleFactory, PortfolioRoleFactory,
ApplicationFactory, UserFactory,
) )
@ -63,23 +68,52 @@ def test_portfolio_audit_log_only_includes_current_portfolio_events():
) )
def test_portfolio_audit_log_includes_app_and_env_events(): def test_get_portfolio_events_includes_app_and_env_events():
# TODO: add in events for
# creating/updating/deleting env_role and app_role
# creating an app_invitation
owner = UserFactory.create() owner = UserFactory.create()
# add portfolio level events
portfolio = PortfolioFactory.create(owner=owner) portfolio = PortfolioFactory.create(owner=owner)
portfolio_events = AuditLog.get_portfolio_events(portfolio)
# add application level events
application = ApplicationFactory.create(portfolio=portfolio) application = ApplicationFactory.create(portfolio=portfolio)
Applications.update(application, {"name": "Star Cruiser"}) Applications.update(application, {"name": "Star Cruiser"})
app_role = ApplicationRoleFactory.create(application=application)
app_invite = ApplicationInvitationFactory.create(role=app_role)
portfolio_and_app_events = AuditLog.get_portfolio_events(portfolio)
assert len(portfolio_events) < len(portfolio_and_app_events)
events = AuditLog.get_portfolio_events(portfolio) # add environment level events
env = EnvironmentFactory.create(application=application)
env_role = EnvironmentRoleFactory.create(environment=env, user=app_role.user)
portfolio_app_and_env_events = AuditLog.get_portfolio_events(portfolio)
assert len(portfolio_and_app_events) < len(portfolio_app_and_env_events)
resource_types = [event.resource_type for event in portfolio_app_and_env_events]
assert "application" in resource_types
assert "application_role" in resource_types
assert "application_invitation" in resource_types
assert "environment" in resource_types
assert "environment_role" in resource_types
def test_get_application_events():
# add in some portfolio level events
portfolio = PortfolioFactory.create()
Portfolios.update(portfolio, {"name": "New Name"})
# add app level events
application = ApplicationFactory.create(portfolio=portfolio)
Applications.update(application, {"name": "Star Cruiser"})
app_role = ApplicationRoleFactory.create(application=application)
app_invite = ApplicationInvitationFactory.create(role=app_role)
env = EnvironmentFactory.create(application=application)
env_role = EnvironmentRoleFactory.create(environment=env, user=app_role.user)
# add rando app
rando_app = ApplicationFactory.create(portfolio=portfolio)
events = AuditLog.get_application_events(application)
for event in events: for event in events:
assert event.portfolio_id == portfolio.id assert event.application_id == application.id
if event.resource_type == 'application': assert not event.application_id == rando_app.id
assert event.application_id == application.id
pass
resource_types = [event.resource_type for event in events]
def test_application_audit_log_does_not_include_portfolio_events(): assert "portfolio" not in resource_types
pass