add officers to task order and redirect to TO when they accept a workspace invite
This commit is contained in:
parent
33de14caaf
commit
d0bfa16f17
38
alembic/versions/71cbe76c3b87_add_officers_to_task_order.py
Normal file
38
alembic/versions/71cbe76c3b87_add_officers_to_task_order.py
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
"""add officers to task order
|
||||||
|
|
||||||
|
Revision ID: 71cbe76c3b87
|
||||||
|
Revises: 6172ac7b8b26
|
||||||
|
Create Date: 2019-01-04 10:16:50.062349
|
||||||
|
|
||||||
|
"""
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
from sqlalchemy.dialects import postgresql
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '71cbe76c3b87'
|
||||||
|
down_revision = '6172ac7b8b26'
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.add_column('task_orders', sa.Column('cor_id', postgresql.UUID(as_uuid=True), nullable=True))
|
||||||
|
op.add_column('task_orders', sa.Column('ko_id', postgresql.UUID(as_uuid=True), nullable=True))
|
||||||
|
op.add_column('task_orders', sa.Column('so_id', postgresql.UUID(as_uuid=True), nullable=True))
|
||||||
|
op.create_foreign_key(None, 'task_orders', 'users', ['so_id'], ['id'])
|
||||||
|
op.create_foreign_key(None, 'task_orders', 'users', ['cor_id'], ['id'])
|
||||||
|
op.create_foreign_key(None, 'task_orders', 'users', ['ko_id'], ['id'])
|
||||||
|
# ### end Alembic commands ###
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
# ### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.drop_constraint(None, 'task_orders', type_='foreignkey')
|
||||||
|
op.drop_constraint(None, 'task_orders', type_='foreignkey')
|
||||||
|
op.drop_constraint(None, 'task_orders', type_='foreignkey')
|
||||||
|
op.drop_column('task_orders', 'so_id')
|
||||||
|
op.drop_column('task_orders', 'ko_id')
|
||||||
|
op.drop_column('task_orders', 'cor_id')
|
||||||
|
# ### end Alembic commands ###
|
@ -4,6 +4,9 @@ from atst.database import db
|
|||||||
from atst.models.task_order import TaskOrder
|
from atst.models.task_order import TaskOrder
|
||||||
from .exceptions import NotFoundError
|
from .exceptions import NotFoundError
|
||||||
|
|
||||||
|
class TaskOrderError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class TaskOrders(object):
|
class TaskOrders(object):
|
||||||
SECTIONS = {
|
SECTIONS = {
|
||||||
@ -89,3 +92,17 @@ class TaskOrders(object):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
OFFICERS = ["contracting_officer", "contracting_officer_representative", "security_officer"]
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def add_officer(cls, task_order, user, role):
|
||||||
|
if role in TaskOrders.OFFICERS:
|
||||||
|
setattr(task_order, role, user)
|
||||||
|
|
||||||
|
db.session.add(task_order)
|
||||||
|
db.session.commit()
|
||||||
|
|
||||||
|
return task_order
|
||||||
|
else:
|
||||||
|
raise TaskOrderError("{} is not an officer role on task orders".format(role))
|
||||||
|
@ -14,7 +14,18 @@ class TaskOrder(Base, mixins.TimestampsMixin):
|
|||||||
workspace = relationship("Workspace")
|
workspace = relationship("Workspace")
|
||||||
|
|
||||||
user_id = Column(ForeignKey("users.id"))
|
user_id = Column(ForeignKey("users.id"))
|
||||||
creator = relationship("User")
|
creator = relationship("User", foreign_keys="TaskOrder.user_id")
|
||||||
|
|
||||||
|
ko_id = Column(ForeignKey("users.id"))
|
||||||
|
contracting_officer = relationship("User", foreign_keys="TaskOrder.ko_id")
|
||||||
|
|
||||||
|
cor_id = Column(ForeignKey("users.id"))
|
||||||
|
contracting_officer_representative = relationship(
|
||||||
|
"User", foreign_keys="TaskOrder.cor_id"
|
||||||
|
)
|
||||||
|
|
||||||
|
so_id = Column(ForeignKey("users.id"))
|
||||||
|
security_officer = relationship("User", foreign_keys="TaskOrder.so_id")
|
||||||
|
|
||||||
scope = Column(String) # Cloud Project Scope
|
scope = Column(String) # Cloud Project Scope
|
||||||
defense_component = Column(String) # Department of Defense Component
|
defense_component = Column(String) # Department of Defense Component
|
||||||
|
@ -154,7 +154,7 @@ class UpdateTaskOrderWorkflow(ShowTaskOrderWorkflow):
|
|||||||
},
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
def _update_invitations(self):
|
def _update_officer_invitations(self):
|
||||||
for officer_type in self.OFFICER_INVITATIONS:
|
for officer_type in self.OFFICER_INVITATIONS:
|
||||||
field = officer_type["field"]
|
field = officer_type["field"]
|
||||||
if (
|
if (
|
||||||
@ -169,16 +169,19 @@ class UpdateTaskOrderWorkflow(ShowTaskOrderWorkflow):
|
|||||||
}
|
}
|
||||||
invite_service = InvitationService(
|
invite_service = InvitationService(
|
||||||
self.user,
|
self.user,
|
||||||
self.task_order.workspace,
|
self.workspace,
|
||||||
{**officer_data, "workspace_role": officer_type["role"]},
|
{**officer_data, "workspace_role": officer_type["role"]},
|
||||||
subject=officer_type["subject"],
|
subject=officer_type["subject"],
|
||||||
email_template=officer_type["template"],
|
email_template=officer_type["template"],
|
||||||
)
|
)
|
||||||
invite_service.invite()
|
invite = invite_service.invite()
|
||||||
|
TaskOrders.add_officer(
|
||||||
|
self.task_order, invite.user, officer_type["role"]
|
||||||
|
)
|
||||||
|
|
||||||
def update(self):
|
def update(self):
|
||||||
self._update_task_order()
|
self._update_task_order()
|
||||||
self._update_invitations()
|
self._update_officer_invitations()
|
||||||
return self.task_order
|
return self.task_order
|
||||||
|
|
||||||
|
|
||||||
|
@ -20,6 +20,25 @@ def send_invite_email(owner_name, token, new_member_email):
|
|||||||
def accept_invitation(token):
|
def accept_invitation(token):
|
||||||
invite = Invitations.accept(g.current_user, token)
|
invite = Invitations.accept(g.current_user, token)
|
||||||
|
|
||||||
|
# TODO: this will eventually redirect to different places depending on
|
||||||
|
# whether the user is an officer for the TO and what kind of officer they
|
||||||
|
# are. It will also have to manage cases like:
|
||||||
|
# - the logged-in user has multiple roles on the TO (e.g., KO and COR)
|
||||||
|
# - the logged-in user has officer roles on multiple unsigned TOs
|
||||||
|
for task_order in invite.workspace.task_orders:
|
||||||
|
if g.current_user == task_order.contracting_officer:
|
||||||
|
return redirect(
|
||||||
|
url_for("task_orders.new", screen=4, task_order_id=task_order.id)
|
||||||
|
)
|
||||||
|
elif g.current_user == task_order.contracting_officer_representative:
|
||||||
|
return redirect(
|
||||||
|
url_for("task_orders.new", screen=4, task_order_id=task_order.id)
|
||||||
|
)
|
||||||
|
elif g.current_user == task_order.security_officer:
|
||||||
|
return redirect(
|
||||||
|
url_for("task_orders.new", screen=4, task_order_id=task_order.id)
|
||||||
|
)
|
||||||
|
|
||||||
return redirect(
|
return redirect(
|
||||||
url_for("workspaces.show_workspace", workspace_id=invite.workspace.id)
|
url_for("workspaces.show_workspace", workspace_id=invite.workspace.id)
|
||||||
)
|
)
|
||||||
|
@ -146,12 +146,19 @@ def test_invite_officers_to_task_order(queue):
|
|||||||
)
|
)
|
||||||
workflow.update()
|
workflow.update()
|
||||||
workspace = task_order.workspace
|
workspace = task_order.workspace
|
||||||
|
# owner and three officers are workspace members
|
||||||
assert len(workspace.members) == 4
|
assert len(workspace.members) == 4
|
||||||
roles = [member.role.name for member in workspace.members]
|
roles = [member.role.name for member in workspace.members]
|
||||||
|
# officers exist in roles
|
||||||
assert "contracting_officer" in roles
|
assert "contracting_officer" in roles
|
||||||
assert "contracting_officer_representative" in roles
|
assert "contracting_officer_representative" in roles
|
||||||
assert "security_officer" in roles
|
assert "security_officer" in roles
|
||||||
|
# email invitations are enqueued
|
||||||
assert len(queue.get_queue()) == 3
|
assert len(queue.get_queue()) == 3
|
||||||
|
# task order has relationship to user for each officer role
|
||||||
|
assert task_order.contracting_officer.dod_id == to_data["ko_dod_id"]
|
||||||
|
assert task_order.contracting_officer_representative.dod_id == to_data["cor_dod_id"]
|
||||||
|
assert task_order.security_officer.dod_id == to_data["so_dod_id"]
|
||||||
|
|
||||||
|
|
||||||
def test_update_does_not_resend_invitation():
|
def test_update_does_not_resend_invitation():
|
||||||
|
@ -6,6 +6,7 @@ from tests.factories import (
|
|||||||
WorkspaceFactory,
|
WorkspaceFactory,
|
||||||
WorkspaceRoleFactory,
|
WorkspaceRoleFactory,
|
||||||
InvitationFactory,
|
InvitationFactory,
|
||||||
|
TaskOrderFactory,
|
||||||
)
|
)
|
||||||
from atst.domain.workspaces import Workspaces
|
from atst.domain.workspaces import Workspaces
|
||||||
from atst.models.workspace_role import Status as WorkspaceRoleStatus
|
from atst.models.workspace_role import Status as WorkspaceRoleStatus
|
||||||
@ -207,3 +208,39 @@ def test_existing_member_invite_resent_to_email_submitted_in_form(
|
|||||||
assert user.email != "example@example.com"
|
assert user.email != "example@example.com"
|
||||||
assert send_mail_job.func.__func__.__name__ == "_send_mail"
|
assert send_mail_job.func.__func__.__name__ == "_send_mail"
|
||||||
assert send_mail_job.args[0] == ["example@example.com"]
|
assert send_mail_job.args[0] == ["example@example.com"]
|
||||||
|
|
||||||
|
|
||||||
|
def test_task_order_officer_accepts_invite(monkeypatch, client, user_session):
|
||||||
|
workspace = WorkspaceFactory.create()
|
||||||
|
task_order = TaskOrderFactory.create(workspace=workspace)
|
||||||
|
user_info = UserFactory.dictionary()
|
||||||
|
|
||||||
|
# create contracting officer
|
||||||
|
user_session(workspace.owner)
|
||||||
|
client.post(
|
||||||
|
url_for("task_orders.new", screen=3, task_order_id=task_order.id),
|
||||||
|
data={
|
||||||
|
"workspace_role": "contracting_officer",
|
||||||
|
"ko_first_name": user_info["first_name"],
|
||||||
|
"ko_last_name": user_info["last_name"],
|
||||||
|
"ko_email": user_info["email"],
|
||||||
|
"ko_dod_id": user_info["dod_id"],
|
||||||
|
"ko_invite": True,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
# contracting officer accepts invitation
|
||||||
|
user = Users.get_by_dod_id(user_info["dod_id"])
|
||||||
|
token = user.invitations[0].token
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"atst.domain.auth.should_redirect_to_user_profile", lambda *args: False
|
||||||
|
)
|
||||||
|
user_session(user)
|
||||||
|
response = client.get(url_for("workspaces.accept_invitation", token=token))
|
||||||
|
|
||||||
|
# user is redirected to the task order review page
|
||||||
|
assert response.status_code == 302
|
||||||
|
to_review_url = url_for(
|
||||||
|
"task_orders.new", screen=4, task_order_id=task_order.id, _external=True
|
||||||
|
)
|
||||||
|
assert response.headers["Location"] == to_review_url
|
||||||
|
Loading…
x
Reference in New Issue
Block a user