Exemple #1
0
class WorkflowTestCase(TestCase):
    def setUp(self):
        super(WorkflowTestCase, self).setUp()
        self.api = Api()
        self.generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
Exemple #2
0
class WorkflowTestCase(TestCase):

  def setUp(self):
    super(WorkflowTestCase, self).setUp()
    self.api = Api()
    self.generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
Exemple #3
0
class WorkflowTestCase(TestCase):

  def setUp(self):
    # old-style class
    TestCase.setUp(self)
    self.api = Api()
    self.generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
Exemple #4
0
class TestOneTimeWorkflowNotification(TestCase):
    """ Tests are defined in the g-sheet test grid under:
    WF EMAILS for unit tests (middle level)
  """
    def setUp(self):
        super(TestOneTimeWorkflowNotification, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.random_people = [
            self.object_generator.generate_person(user_role="Administrator")[1]
            for _ in range(5)
        ]
        self.create_test_cases()

        self.create_users()

        db.session.query(Notification).delete()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    def tearDown(self):
        db.session.query(Notification).delete()

    def short_dict(self, obj, plural):
        return {
            "href": "/api/%s/%d" % (plural, obj.id),
            "id": obj.id,
            "type": obj.__class__.__name__,
        }

    def setup_cycle_tasks(self):
        """Prepare environment with couple of active cycle tasks."""
        with freeze_time("2018-11-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)
            self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)
        return all_models.CycleTaskGroupObjectTask.query

    def assert_nofication_sent_with(self, text):
        """Assert if text exists in sent notification."""
        with mock.patch("ggrc.notifications.common.send_email") as send_email:
            self.client.get("/_notifications/send_daily_digest")
            _, _, content = send_email.call_args[0]
        self.assertIn(text, content)

    def assert_nofication_sent_without(self, text):
        """Assert if text doesn't exist in sent notification."""
        with mock.patch("ggrc.notifications.common.send_email") as send_email:
            self.client.get("/_notifications/send_daily_digest")
            _, _, content = send_email.call_args[0]
        self.assertNotIn(text, content)

    def test_one_time_wf(self):
        # setup
        with freeze_time("2015-04-07 03:21:34"):
            wf_response, wf = self.wf_generator.generate_workflow(
                data={
                    # admin will be the current user
                    "notify_on_change":
                    True,  # force real time updates
                    "title":
                    "One-time WF",
                    "notify_custom_message":
                    textwrap.dedent("""\
              Hi all.
              Did you know that Irelnd city namd Newtownmountkennedy has 19
              letters? But it's not the longest one. The recordsman is the
              city in New Zealand that contains 97 letter."""),
                })

            _, tg = self.wf_generator.generate_task_group(
                wf,
                data={
                    "title": "TG #1 for the One-time WF",
                    "contact": self.short_dict(self.tgassignee1, "people"),
                })

            self.wf_generator.generate_task_group_task(
                tg, {
                    "title": "task #1 for one-time workflow",
                    "contact": self.short_dict(self.member1, "people"),
                    "start_date": "04/07/2015",
                    "end_date": "04/15/2015",
                })

            self.wf_generator.generate_task_group_object(
                tg, self.random_objects[0])
            self.wf_generator.generate_task_group_object(
                tg, self.random_objects[1])

        # test
        with freeze_time("2015-04-07 03:21:34"):
            cycle_response, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

            common.get_daily_notifications()

    def test_deprecated_ct_acl_update(self):
        """Test if acl update for deprecated CT will not create notification."""
        cycle_task = self.setup_cycle_tasks().first()
        cycle_task_title = cycle_task.title
        response = self.api.put(cycle_task, {"status": "Deprecated"})
        self.assert200(response)
        self.assert_nofication_sent_without(cycle_task_title)

        task_assignee = all_models.AccessControlRole.query.filter_by(
            name="Task Assignees",
            object_type="CycleTaskGroupObjectTask",
        ).first()
        person = self.object_generator.generate_person(user_role="Creator")[1]
        response = self.api.put(
            cycle_task, {
                "access_control_list": [{
                    "ac_role_id": task_assignee.id,
                    "person": {
                        "id": person.id,
                        "type": "Person",
                    }
                }]
            })
        self.assert200(response)
        self.assert_nofication_sent_without(cycle_task_title)

    def test_restore_deprecated_ct(self):
        """Test notifications for CT which was restored from Deprecated."""
        cycle_task = self.setup_cycle_tasks().first()
        cycle_task_title = cycle_task.title

        self.assert_nofication_sent_with(cycle_task_title)

        response = self.api.put(cycle_task, {"status": "Deprecated"})
        self.assert200(response)
        self.assert_nofication_sent_without(cycle_task_title)

        response = self.api.put(cycle_task, {"status": "Assigned"})
        self.assert200(response)
        self.assert_nofication_sent_with(cycle_task_title)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "description":
            "some test workflow",
            # admin will be current user with id == 1
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "task_group_tasks": [
                        {
                            "title": "task_{}".format(str(uuid.uuid4())),
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        },
                        {
                            "title": "task_{}".format(str(uuid.uuid4())),
                            "description": "some task",
                            "contact": person_dict(self.random_people[1].id),
                            "start_date": date(2015, 5, 4),
                            "end_date": date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "task_group_tasks": [
                        {
                            "title": "task_{}".format(str(uuid.uuid4())),
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 8),  # friday
                            "end_date": date(2015, 5, 12),
                        },
                        {
                            "title": "task_{}".format(str(uuid.uuid4())),
                            "description": "some task",
                            "contact": person_dict(self.random_people[2].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }

    def create_users(self):
        _, self.admin1 = self.object_generator.generate_person(
            # data={"name": "User1 Admin1", "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.tgassignee1 = self.object_generator.generate_person(
            # data={"name": "User2 TGassignee1",
            #       "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.member1 = self.object_generator.generate_person(
            # data={"name": "User3 Member1", "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.member2 = self.object_generator.generate_person(
            # data={"name": "User4 Member2", "email": "*****@*****.**"},
            user_role="Administrator")
Exemple #5
0
class TestCycleTaskStatusChange(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        super(TestCycleTaskStatusChange, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        all_models.Notification.query.delete()
        self.random_objects = self.object_generator.generate_random_objects(2)
        self.user = self.create_user_with_role(role="Administrator")
        self.secondary_assignee = self.create_user_with_role(role="Reader")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        all_models.Notification.__init__ = init_decorator(
            all_models.Notification.__init__)

    def test_task_declined_notification_created(self):
        """Test declined cycle task notifications"""
        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Declined")

            notif = self.get_notifications_by_type(task1,
                                                   "cycle_task_declined")
            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

    def test_all_tasks_finished_notification_created(self):
        """Test all task completed notifications"""
        with freeze_time("2015-05-01 13:20:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            generate_cycle_tasks_notifs()

            notif = self.get_notifications_by_type(
                cycle, "all_cycle_tasks_completed")
            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

            notif = self.get_notifications_by_type(
                task1, "all_cycle_tasks_completed")
            self.assertEqual(notif, [])

    def test_multi_all_tasks_finished_notification_created(self):
        """Test several all task completed notifications"""

        with freeze_time("2015-05-01 13:20:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_2)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            generate_cycle_tasks_notifs()
            notif = self.get_notifications_by_type(
                cycle, "all_cycle_tasks_completed")

            # there is still one task in the cycle, so there should be no
            # notifications for all tasks completed
            self.assertEqual(notif, [])

            notif = self.get_notifications_by_type(
                task1, "all_cycle_tasks_completed")

            # The task was verified, so there should be no notifications left for due
            # dates.
            self.assertEqual(notif, [])

            task2 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[1].id)

            self.task_change_status(task2)

            generate_cycle_tasks_notifs()
            notif = self.get_notifications_by_type(
                cycle, "all_cycle_tasks_completed")

            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

    @patch("ggrc.notifications.common.send_email")
    def test_single_task_declined(self, mock_mail):
        """Test moving the end date to the future on declined task.

    It is done before due_in and due_today notifications have been sent.
    """

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Finished")

            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Declined")

            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertIn("task_declined", notif_data[user.email])

    @patch("ggrc.notifications.common.send_email")
    def test_single_task_accepted(self, mock_mail):
        """Test moving the end date to the future on accepted task.

    It is done before due_in and due_today notifications have been sent.
    """

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Finished")

            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time("2015-05-03 13:20:34"):
            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            generate_cycle_tasks_notifs()
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertNotIn(user.email, notif_data)
            self.assertIn("all_tasks_completed",
                          notif_data["*****@*****.**"])

    @patch("ggrc.notifications.common.send_email")
    def test_end_cycle(self, mock_mail):
        """Manually ending cycle should stop all notifications for that cycle."""

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)
            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time("2015-05-03"):
            _, notif_data = common.get_daily_notifications()
            cycle = Cycle.query.get(cycle.id)
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
            self.wf_generator.modify_object(cycle, data={"is_current": False})
            cycle = Cycle.query.get(cycle.id)
            self.assertFalse(cycle.is_current)

            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertNotIn(user.email, notif_data)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        task_assignee_role_id = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        task_secondary_assignee = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Secondary Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "is_verification_needed":
            True,
            # admin will be current user with id == 1
            "task_groups": [{
                "title":
                "single task group",
                "contact":
                person_dict(self.user.id),
                "task_group_tasks": [{
                    "title":
                    "task 1",
                    "description":
                    "single task in a wf",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2015, 5, 1),  # friday
                    "end_date":
                    date(2015, 5, 5),
                }],
            }]
        }

        self.one_time_workflow_2 = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "is_verification_needed":
            True,
            # admin will be current user with id == 1
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.user.id),
                "task_group_tasks": [
                    {
                        "title":
                        "task 1",
                        "description":
                        "two taks in wf with different objects",
                        "access_control_list": [
                            acl_helper.get_acl_json(task_assignee_role_id,
                                                    self.user.id)
                        ],
                        "start_date":
                        date(2015, 5, 1),  # friday
                        "end_date":
                        date(2015, 5, 5),
                    },
                    {
                        "title":
                        "task 2",
                        "description":
                        "two taks in wf with different objects",
                        "access_control_list": [
                            acl_helper.get_acl_json(task_assignee_role_id,
                                                    self.user.id)
                        ],
                        "start_date":
                        date(2015, 5, 1),  # friday
                        "end_date":
                        date(2015, 5, 5),
                    }
                ],
                "task_group_objects":
                self.random_objects
            }]
        }

    def get_notification_type(self, name):
        return all_models.NotificationType.query.filter(
            all_models.NotificationType.name == name).one()

    def task_change_status(self, task, status="Verified"):
        self.wf_generator.modify_object(task, data={"status": status})

        task = CycleTaskGroupObjectTask.query.get(task.id)

        self.assertEqual(task.status, status)

    def get_notifications_by_type(self, obj, notif_type):
        return all_models.Notification.query.filter(
            and_(
                all_models.Notification.object_id == obj.id,
                all_models.Notification.object_type == obj.type,
                all_models.Notification.sent_at == None,  # noqa
                all_models.Notification.notification_type ==
                self.get_notification_type(notif_type))).all()
class TestCycleTaskImportUpdate(BaseTestCycleTaskImportUpdate):

  """ This class contains simple cycle task update tests using import
  functionality
  """

  CSV_DIR = join(abspath(dirname(__file__)), "test_csvs/")

  def setUp(self):
    super(TestCycleTaskImportUpdate, self).setUp()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.person_1 = self.object_generator.generate_person(
        user_role="Administrator")
    self.ftime_active = "2016-07-01"
    self.ftime_historical = "2014-05-01"
    self._create_test_cases_data()
    # It is needed because cycle-tasks are generated automatically with
    # 'slug' based on auto_increment 'id' field.
    # At start of each test we suppose that created cycle-task's 'slug'
    # lie in range from 1 to 10.
    db.session.execute('ALTER TABLE cycle_task_group_object_tasks '
                       'AUTO_INCREMENT = 1')

  def test_cycle_task_correct(self):
    """Test cycle task update via import with correct data"""
    self._generate_cycle_tasks()
    with freeze_time(self.ftime_active):
      response = self.import_file("cycle_task_correct.csv")
      self._check_csv_response(response, {})
      self._cmp_tasks(self.expected_cycle_task_correct)

  def test_cycle_task_warnings(self):
    """Test cycle task update via import with data which is the reason of
    warnings about non-importable columns."""
    self._generate_cycle_tasks()
    with freeze_time(self.ftime_active):
      response = self.import_file("cycle_task_warnings.csv")
      self._check_csv_response(response, self.expected_warnings)
      self._cmp_tasks(self.expected_cycle_task_correct)

  def test_cycle_task_create_error(self):
    """Test cycle task update via import with data which is the reason of
    errors about new cycle task creation."""
    self._generate_cycle_tasks()
    with freeze_time(self.ftime_active):
      response = self.import_file("cycle_task_create_error.csv")
      self._check_csv_response(response, self.expected_create_error)
      self._cmp_tasks(self.expected_cycle_task_correct)

  def test_cycle_task_date_error(self):
    """Test cycle task update via import with data which is the reason of
    errors about incorrect dates in csv file."""
    self._generate_cycle_tasks()
    with freeze_time(self.ftime_active):
      response = self.import_file("cycle_task_date_error.csv")
      self._check_csv_response(response, self.expected_date_error)
      self._cmp_tasks(self.expected_cycle_task_date_error)

  def test_cycle_task_permission_error(self):
    """Test cycle task update via import with non-admin user which is the
    reason of error. Only admin can update cycle tasks via import."""
    self._generate_cycle_tasks()
    with freeze_time(self.ftime_active):
      _, creator = self.object_generator.generate_person(user_role="Creator")
      response = self.import_file("cycle_task_correct.csv", person=creator)
      self._check_csv_response(response, self.expected_permission_error)
      # Cycle tasks' data shouldn't be changed in test DB after import run from
      # non-admin user
      expected_cycle_task_permission_error = {}
      expected_cycle_task_permission_error.update(
          self.generated_cycle_tasks_active)
      expected_cycle_task_permission_error.update(
          self.generated_cycle_tasks_historical)
      self._cmp_tasks(expected_cycle_task_permission_error)

  def _cmp_tasks(self, expected_ctasks):
    """Compare tasks values from argument's list and test DB."""
    for ctask in db.session.query(CycleTaskGroupObjectTask).all():
      if ctask.slug not in expected_ctasks:
        continue
      exp_task = expected_ctasks[ctask.slug]
      for attr, val in exp_task.iteritems():
        self.assertEqual(str(getattr(ctask, attr, None)), val)

  # pylint: disable=too-many-arguments
  def _activate_workflow(self, ftime, workflow, task_group, task_group_tasks,
                         random_object, cycle_tasks):
    """Helper which is responsible for active cycle-tasks creation"""
    with freeze_time(ftime):
      _, wf = self.wf_generator.generate_workflow(workflow)
      _, tg = self.wf_generator.generate_task_group(wf, task_group)
      for task in task_group_tasks:
        self.wf_generator.generate_task_group_task(tg, task)
      self.wf_generator.generate_task_group_object(tg, random_object)

      _, cycle = self.wf_generator.generate_cycle(wf)
      self.wf_generator.activate_workflow(wf)

      for exp_slug, exp_task in cycle_tasks.iteritems():
        obj = db.session.query(CycleTaskGroupObjectTask).filter_by(
            slug=exp_slug
        ).first()
        if exp_task["status"] == "Verified":
          self.wf_generator.modify_object(obj, {"status": "Finished"})
        self.wf_generator.modify_object(obj, {"status": exp_task["status"]})
    self._cmp_tasks(cycle_tasks)
    return cycle

  def _generate_cycle_tasks(self):
    """Helper which is responsible for test data creation"""
    self._activate_workflow(self.ftime_active, self.workflow_active,
                            self.task_group_active,
                            self.task_group_tasks_active,
                            self.random_objects[0],
                            self.generated_cycle_tasks_active)
    cycle = self._activate_workflow(self.ftime_historical,
                                    self.workflow_historical,
                                    self.task_group_historical,
                                    self.task_group_tasks_historical,
                                    self.random_objects[1],
                                    self.generated_cycle_tasks_historical)
    with freeze_time(self.ftime_historical):
      cycle = Cycle.query.get(cycle.id)
      self.wf_generator.modify_object(cycle, data={"is_current": False})

  def _create_test_cases_data(self):
    """Create test cases data: for object generation,
    expected data for checks"""
    def person_dict(person_id):
      """Return person data"""
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.workflow_active = {
        "title": "workflow active title",
        "description": "workflow active description",
        "frequency": "one_time",
        "owners": [person_dict(self.person_1.id)],
        "notify_on_change": False,
    }

    self.task_group_active = {
        "title": "task group active title",
        "contact": person_dict(self.person_1.id),
    }

    self.task_group_tasks_active = [{
        "title": "task active title 1",
        "description": "task active description 1",
        "contact": person_dict(self.person_1.id),
        "start_date": "07/01/2016",
        "end_date": "07/06/2016",
    }, {
        "title": "task active title 2",
        "description": "task active description 2",
        "contact": person_dict(self.person_1.id),
        "start_date": "07/07/2016",
        "end_date": "07/12/2016",
    }, {
        "title": "task active title 3",
        "description": "task active description 3",
        "contact": person_dict(self.person_1.id),
        "start_date": "07/13/2016",
        "end_date": "07/18/2016",
    }, {
        "title": "task active title 4",
        "description": "task active description 4",
        "contact": person_dict(self.person_1.id),
        "start_date": "07/19/2016",
        "end_date": "07/24/2016",
    }, {
        "title": "task active title 5",
        "description": "task active description 5",
        "contact": person_dict(self.person_1.id),
        "start_date": "07/25/2016",
        "end_date": "07/30/2016",
    }]

    # Active cycle tasks which should be generated from previous structure
    # at the beginning of each test
    self.generated_cycle_tasks_active = {
        "CYCLETASK-1": {
            "title": self.task_group_tasks_active[0]["title"],
            "description": self.task_group_tasks_active[0]["description"],
            "start_date": "2016-07-01",
            "end_date": "2016-07-06",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Assigned"
        },
        "CYCLETASK-2": {
            "title": self.task_group_tasks_active[1]["title"],
            "description": self.task_group_tasks_active[1]["description"],
            "start_date": "2016-07-07",
            "end_date": "2016-07-12",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Declined"
        },
        "CYCLETASK-3": {
            "title": self.task_group_tasks_active[2]["title"],
            "description": self.task_group_tasks_active[2]["description"],
            "start_date": "2016-07-13",
            "end_date": "2016-07-18",
            "finished_date": "None",
            "verified_date": "None",
            "status": "InProgress"
        },
        "CYCLETASK-4": {
            "title": self.task_group_tasks_active[3]["title"],
            "description": self.task_group_tasks_active[3]["description"],
            "start_date": "2016-07-19",
            "end_date": "2016-07-24",
            "finished_date": "2016-07-01 00:00:00",
            "verified_date": "None",
            "status": "Finished"
        },
        "CYCLETASK-5": {
            "title": self.task_group_tasks_active[4]["title"],
            "description": self.task_group_tasks_active[4]["description"],
            "start_date": "2016-07-25",
            "end_date": "2016-07-30",
            "finished_date": "2016-07-01 00:00:00",
            "verified_date": "2016-07-01 00:00:00",
            "status": "Verified"
        }
    }

    self.workflow_historical = {
        "title": "workflow historical title",
        "description": "workflow historical description",
        "frequency": "one_time",
        "owners": [person_dict(self.person_1.id)],
        "notify_on_change": False,
    }

    self.task_group_historical = {
        "title": "task group historical title",
        "contact": person_dict(self.person_1.id),
    }

    self.task_group_tasks_historical = [{
        "title": "task historical title 1",
        "description": "task historical description 1",
        "contact": person_dict(self.person_1.id),
        "start_date": "05/01/2014",
        "end_date": "05/06/2014",
    }, {
        "title": "task historical title 2",
        "description": "task historical description 2",
        "contact": person_dict(self.person_1.id),
        "start_date": "05/07/2014",
        "end_date": "05/12/2014",
    }, {
        "title": "task historical title 3",
        "description": "task historical description 3",
        "contact": person_dict(self.person_1.id),
        "start_date": "05/13/2014",
        "end_date": "05/18/2014",
    }, {
        "title": "task historical title 4",
        "description": "task historical description 4",
        "contact": person_dict(self.person_1.id),
        "start_date": "05/19/2014",
        "end_date": "05/24/2014",
    }, {
        "title": "task historical title 5",
        "description": "task historical description 5",
        "contact": person_dict(self.person_1.id),
        "start_date": "05/25/2014",
        "end_date": "05/30/2014",
    },
    ]

    # Historical cycle tasks which should be generated from previous structure
    # at the beginning of each test.
    self.generated_cycle_tasks_historical = {
        "CYCLETASK-6": {
            "title": self.task_group_tasks_historical[0]["title"],
            "description": self.task_group_tasks_historical[0]["description"],
            "start_date": "2014-05-01",
            "end_date": "2014-05-06",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Assigned"
        },
        "CYCLETASK-7": {
            "title": self.task_group_tasks_historical[1]["title"],
            "description": self.task_group_tasks_historical[1]["description"],
            "start_date": "2014-05-07",
            "end_date": "2014-05-12",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Declined"
        },
        "CYCLETASK-8": {
            "title": self.task_group_tasks_historical[2]["title"],
            "description": self.task_group_tasks_historical[2]["description"],
            "start_date": "2014-05-13",
            "end_date": "2014-05-18",
            "finished_date": "None",
            "verified_date": "None",
            "status": "InProgress"
        },
        "CYCLETASK-9": {
            "title": self.task_group_tasks_historical[3]["title"],
            "description": self.task_group_tasks_historical[3]["description"],
            "start_date": "2014-05-19",
            "end_date": "2014-05-24",
            "finished_date": "2014-05-01 00:00:00",
            "verified_date": "None",
            "status": "Finished"
        },
        "CYCLETASK-10": {
            "title": self.task_group_tasks_historical[4]["title"],
            "description": self.task_group_tasks_historical[4]["description"],
            "start_date": "2014-05-25",
            "end_date": "2014-05-30",
            "finished_date": "2014-05-01 00:00:00",
            "verified_date": "2014-05-01 00:00:00",
            "status": "Verified"
        }
    }

    # Expected cycle tasks which should be created in correct cycle task update
    # case. It is needed for most tests.
    self.expected_cycle_task_correct = {
        "CYCLETASK-1": {
            "title": self.task_group_tasks_active[0]["title"] + " one",
            "description":
                self.task_group_tasks_active[0]["description"] + " one",
            "start_date": "2016-06-01",
            "end_date": "2016-06-06",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Assigned"
        },
        "CYCLETASK-2": {
            "title": self.task_group_tasks_active[1]["title"] + " two",
            "description":
                self.task_group_tasks_active[1]["description"] + " two",
            "start_date": "2016-06-07",
            "end_date": "2016-06-12",
            "finished_date": "None",
            "verified_date": "None",
            "status": "Declined"
        },
        "CYCLETASK-3": {
            "title": self.task_group_tasks_active[2]["title"] + " three",
            "description":
                self.task_group_tasks_active[2]["description"] + " three",
            "start_date": "2016-06-13",
            "end_date": "2016-06-18",
            "finished_date": "None",
            "verified_date": "None",
            "status": "InProgress"
        },
        "CYCLETASK-4": {
            "title": self.task_group_tasks_active[3]["title"] + " four",
            "description":
                self.task_group_tasks_active[3]["description"] + " four",
            "start_date": "2016-06-19",
            "end_date": "2016-06-24",
            "finished_date": "2016-07-19 00:00:00",
            "verified_date": "None",
            "status": "Finished"
        },
        "CYCLETASK-5": {
            "title": self.task_group_tasks_active[4]["title"] + " five",
            "description":
                self.task_group_tasks_active[4]["description"] + " five",
            "start_date": "2016-06-25",
            "end_date": "2016-06-30",
            "finished_date": "2016-07-25 00:00:00",
            "verified_date": "2016-08-30 00:00:00",
            "status": "Verified"
        },
        "CYCLETASK-6": {
            "title": self.task_group_tasks_historical[0]["title"] + " one",
            "description":
                self.task_group_tasks_historical[0]["description"] + " one",
            "start_date": "2014-04-01",
            "end_date": "2014-04-06",
            "finished_date": "2014-05-01 00:00:00",
            "verified_date": "2014-06-06 00:00:00",
            "status": "Assigned"
        },
        "CYCLETASK-7": {
            "title": self.task_group_tasks_historical[1]["title"] + " two",
            "description":
                self.task_group_tasks_historical[1]["description"] + " two",
            "start_date": "2014-04-07",
            "end_date": "2014-04-12",
            "finished_date": "2014-05-07 00:00:00",
            "verified_date": "None",
            "status": "Declined"
        },
        "CYCLETASK-8": {
            "title": self.task_group_tasks_historical[2]["title"] + " three",
            "description":
                self.task_group_tasks_historical[2]["description"] + " three",
            "start_date": "2014-04-13",
            "end_date": "2014-04-18",
            "finished_date": "2014-05-13 00:00:00",
            "verified_date": "2014-06-18 00:00:00",
            "status": "InProgress"
        },
        "CYCLETASK-9": {
            "title": self.task_group_tasks_historical[3]["title"] + " four",
            "description":
                self.task_group_tasks_historical[3]["description"] + " four",
            "start_date": "2014-04-19",
            "end_date": "2014-04-24",
            "finished_date": "2014-05-19 00:00:00",
            "verified_date": "None",
            "status": "Finished"
        },
        "CYCLETASK-10": {
            "title": self.task_group_tasks_historical[4]["title"] + " five",
            "description":
                self.task_group_tasks_historical[4]["description"] + " five",
            "start_date": "2014-04-25",
            "end_date": "2014-04-30",
            "finished_date": "2014-05-25 00:00:00",
            "verified_date": "2014-06-30 00:00:00",
            "status": "Verified"
        }
    }

    # Below is description of warning for non-importable columns. It is needed
    # for test_cycle_task_warnings.
    importable_column_names = []
    for field_name in CycleTaskGroupObjectTask.IMPORTABLE_FIELDS:
      if field_name == 'slug':
        continue
      # pylint: disable=protected-access
      name = CycleTaskGroupObjectTask._aliases.get(field_name, field_name)
      if isinstance(name, dict):
        name = name['display_name']
      importable_column_names.append(name)
    self.expected_warnings = self.generate_expected_warning(
        *importable_column_names)

    # This is an error message which should be shown during
    # test_cycle_task_create_error test
    self.expected_create_error = {
        'Cycle Task': {
            'row_errors': {errors.CREATE_INSTANCE_ERROR.format(line=13)}
        }
    }

    # Below is expected date errors for test_cycle_task_date_error. They should
    # be shown during date validator's tests.
    self.expected_date_error = {
        'Cycle Task': {
            'row_errors': {
                errors.INVALID_START_END_DATES.format(
                    line=3,
                    start_date="Start Date",
                    end_date="End Date",
                ),
                errors.INVALID_STATUS_DATE_CORRELATION.format(
                    line=4,
                    date="Actual Finish Date",
                    status="not Finished",
                ),
                errors.INVALID_STATUS_DATE_CORRELATION.format(
                    line=5,
                    date="Actual Verified Date",
                    status="not Verified",
                ),
                errors.INVALID_STATUS_DATE_CORRELATION.format(
                    line=6,
                    date="Actual Verified Date",
                    status="not Verified",
                ),
                errors.INVALID_START_END_DATES.format(
                    line=7,
                    start_date="Actual Finish Date",
                    end_date="Actual Verified Date",
                ),
                errors.INVALID_START_END_DATES.format(
                    line=8,
                    start_date="Start Date",
                    end_date="End Date",
                ),
                errors.MISSING_VALUE_ERROR.format(
                    line=9,
                    column_name="Actual Finish Date",
                ),
                errors.INVALID_START_END_DATES.format(
                    line=10,
                    start_date="Actual Finish Date",
                    end_date="Actual Verified Date",
                ),
            },
        }
    }
    # Below is expected cycle-tasks data which should appear in test DB after
    # test_cycle_task_date_error run
    self.expected_cycle_task_date_error = dict()
    self.expected_cycle_task_date_error.update(
        self.generated_cycle_tasks_active)
    self.expected_cycle_task_date_error.update(
        self.generated_cycle_tasks_historical)
    self.expected_cycle_task_date_error["CYCLETASK-9"] = (
        self.expected_cycle_task_correct["CYCLETASK-9"])
    self.expected_cycle_task_date_error["CYCLETASK-10"] = (
        self.expected_cycle_task_correct["CYCLETASK-10"])

    # Expected error message which should be shown after
    # test_cycle_task_permission_error run
    self.expected_permission_error = {
        'Cycle Task': {
            'block_errors': {errors.PERMISSION_ERROR.format(line=2)}
        }
    }
class TestBasicWorkflowActions(TestCase):
    """
  Tests for basic workflow actions
  """
    def setUp(self):
        super(TestBasicWorkflowActions, self).setUp()
        self.api = Api()
        self.generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.create_test_cases()

    def tearDown(self):
        pass

    def test_create_workflows(self):
        _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
        self.assertIsInstance(wflow, Workflow)

        task_groups = db.session.query(TaskGroup)\
            .filter(TaskGroup.workflow_id == wflow.id).all()

        self.assertEqual(len(task_groups),
                         len(self.one_time_workflow_1["task_groups"]))

    def test_workflows(self):
        for workflow in self.all_workflows:
            _, wflow = self.generator.generate_workflow(workflow)
            self.assertIsInstance(wflow, Workflow)

            task_groups = db.session.query(TaskGroup)\
                .filter(TaskGroup.workflow_id == wflow.id).all()

            self.assertEqual(len(task_groups), len(workflow["task_groups"]))

    def test_activate_wf(self):
        for workflow in self.all_workflows:
            _, wflow = self.generator.generate_workflow(workflow)
            response, wflow = self.generator.activate_workflow(wflow)

            self.assert200(response)

    def test_one_time_workflow_edits(self):
        _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)

        wf_dict = {"title": "modified one time wf"}
        self.generator.modify_workflow(wflow, data=wf_dict)

        modified_wf = db.session.query(Workflow).filter(
            Workflow.id == wflow.id).one()
        self.assertEqual(wf_dict["title"], modified_wf.title)

    def test_one_time_wf_activate(self):
        _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
        self.generator.generate_cycle(wflow)
        self.generator.activate_workflow(wflow)

        tasks = [
            len(tg.get("task_group_tasks", []))
            for tg in self.one_time_workflow_1["task_groups"]
        ]

        cycle_tasks = db.session.query(CycleTaskGroupObjectTask).join(
            Cycle).join(Workflow).filter(Workflow.id == wflow.id).all()
        active_wf = db.session.query(Workflow).filter(
            Workflow.id == wflow.id).one()

        self.assertEqual(sum(tasks), len(cycle_tasks))
        self.assertEqual(active_wf.status, "Active")

    def test_one_time_wf_state_transition_dates(self):
        _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
        self.generator.generate_cycle(wflow)
        self.generator.activate_workflow(wflow)

        cycle_tasks = db.session.query(CycleTaskGroupObjectTask).join(
            Cycle).join(Workflow).filter(Workflow.id == wflow.id).all()
        with freeze_time("2015-6-9"):
            today = dtm.date.today()
            transitions = [
                ("In Progress", None, None),
                ("Finished", today, None),
                ("Declined", None, None),
                ("Finished", today, None),
                ("Verified", today, today),
                ("Finished", today, None),
            ]
            # Iterate over possible transitions and check if dates got set correctly
            for (status, expected_finished, expected_verified) in transitions:
                cycle_task = cycle_tasks[0]
                _, task = self.generator.modify_object(cycle_task,
                                                       {"status": status})
                self.assertEqual(task.finished_date, expected_finished)
                self.assertEqual(task.verified_date, expected_verified)

    def test_delete_calls(self):
        _, workflow = self.generator.generate_workflow()
        self.generator.generate_task_group(workflow)
        _, task_group = self.generator.generate_task_group(workflow)
        task_groups = db.session.query(TaskGroup).filter(
            TaskGroup.workflow_id == workflow.id).all()
        self.assertEqual(len(task_groups), 2)

        response = self.generator.api.delete(task_group)
        self.assert200(response)

        task_groups = db.session.query(TaskGroup).filter(
            TaskGroup.workflow_id == workflow.id).all()
        self.assertEqual(len(task_groups), 1)

    def create_test_cases(self):

        self.quarterly_wf_1 = {
            "title":
            "quarterly wf 1",
            "description":
            "",
            "unit":
            "month",
            "repeat_every":
            3,
            "task_groups": [
                {
                    "title":
                    "tg_1",
                    "task_group_tasks": [
                        {
                            "description": factories.random_str(100),
                        },
                        {
                            "description": factories.random_str(100),
                        },
                        {
                            "description": factories.random_str(100),
                        },
                    ],
                },
            ]
        }

        self.weekly_wf_1 = {
            "title":
            "weekly thingy",
            "description":
            "start this many a time",
            "unit":
            "week",
            "repeat_every":
            1,
            "task_groups": [
                {
                    "title":
                    "tg_2",
                    "task_group_tasks": [
                        {
                            "description": factories.random_str(100),
                        },
                        {
                            "title": "monday task",
                        },
                        {
                            "title": "weekend task",
                        },
                    ],
                    "task_group_objects":
                    self.random_objects
                },
            ]
        }

        self.one_time_workflow_1 = {
            "title":
            "one time wf test",
            "description":
            "some test workflow",
            "is_verification_needed":
            True,
            "task_groups": [{
                "title": "tg_1",
                "task_group_tasks": [{}, {}, {}]
            }, {
                "title":
                "tg_2",
                "task_group_tasks": [{
                    "description": factories.random_str(100)
                }, {}],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "tg_3",
                "task_group_tasks": [{
                    "title": "simple task 1",
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }],
                "task_group_objects":
                self.random_objects
            }]
        }
        self.one_time_workflow_2 = {
            "title":
            "test_wf_title",
            "description":
            "some test workflow",
            "task_groups": [{
                "title": "tg_1",
                "task_group_tasks": [{}, {}, {}]
            }, {
                "title":
                "tg_2",
                "task_group_tasks": [{
                    "description": factories.random_str(100)
                }, {}],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "tg_3",
                "task_group_tasks": [{
                    "title": "simple task 1",
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }],
                "task_group_objects": []
            }]
        }

        self.monthly_workflow_1 = {
            "title":
            "monthly test wf",
            "description":
            "start this many a time",
            "unit":
            "month",
            "repeat_every":
            1,
            "task_groups": [
                {
                    "title":
                    "tg_2",
                    "task_group_tasks": [{
                        "description":
                        factories.random_str(100),
                    }, {
                        "title": "monday task",
                    }, {
                        "title": "weekend task",
                    }],
                    "task_group_objects":
                    self.random_objects
                },
            ]
        }
        self.all_workflows = [
            self.one_time_workflow_1,
            self.one_time_workflow_2,
            self.weekly_wf_1,
            self.monthly_workflow_1,
            self.quarterly_wf_1,
        ]
class TestTaskDueNotifications(TestNotifications):
    """Test suite for task due soon/today notifications."""
    def setUp(self):
        super(TestTaskDueNotifications, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        all_models.Notification.query.delete()

        self._fix_notification_init()

        self.random_objects = self.object_generator.generate_random_objects(2)
        self.user = self.create_user_with_role(role="Administrator")
        self.secondary_assignee = self.create_user_with_role(role="Reader")

        task_assignee_role_id = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        task_secondary_assignee = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Secondary Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        self.one_time_workflow = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "is_verification_needed":
            False,
            # admin will be current user with id == 1
            "task_groups": [{
                "title":
                "one time task group",
                "contact": {
                    "href": "/api/people/{}".format(self.user.id),
                    "id": self.user.id,
                    "type": "Person",
                },
                "task_group_tasks": [{
                    "title":
                    "task 1",
                    "description":
                    "some task",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2017, 5, 15),
                    "end_date":
                    date(2017, 6, 11),
                }, {
                    "title":
                    "task 2",
                    "description":
                    "some task 2",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2017, 5, 8),
                    "end_date":
                    date(2017, 6, 12),
                }, {
                    "title":
                    "task 3",
                    "description":
                    "some task 3",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2017, 5, 31),
                    "end_date":
                    date(2017, 6, 13),
                }, {
                    "title":
                    "task 4",
                    "description":
                    "some task 4",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2017, 6, 2),
                    "end_date":
                    date(2017, 6, 14),
                }, {
                    "title":
                    "task 5",
                    "description":
                    "some task 5",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_assignee_role_id,
                                                self.user.id),
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "start_date":
                    date(2017, 6, 8),
                    "end_date":
                    date(2017, 6, 15),
                }],
                "task_group_objects":
                self.random_objects
            }]
        }

    @ddt.unpack
    @ddt.data(
        ("2017-06-12 12:12:12", ["task 1"], ["task 2"], ["task 3"]),
        ("2017-06-13 13:13:13", ["task 1", "task 2"], ["task 3"], ["task 4"]),
    )
    @patch("ggrc.notifications.common.send_email")
    def test_creating_obsolete_notifications(self, fake_now, expected_overdue,
                                             expected_due_today,
                                             expected_due_in, _):
        """Notifications already obsolete on creation date should not be created.
    """
        # pylint: disable=invalid-name
        with freeze_time("2017-06-12 09:39:32"):
            tmp = self.one_time_workflow.copy()
            _, workflow = self.wf_generator.generate_workflow(tmp)
            self.wf_generator.generate_cycle(workflow)
            response, workflow = self.wf_generator.activate_workflow(workflow)
            self.assert200(response)

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time(fake_now):
            # mark all yeasterday notifications as sent
            all_models.Notification.query.filter(
                sa.func.DATE(all_models.Notification.send_on) < date.today()
            ).update(
                {
                    all_models.Notification.sent_at:
                    datetime.now() - timedelta(1)
                },
                synchronize_session="fetch")

            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                user_notifs = notif_data.get(user.email, {})

                actual_overdue = [
                    n['title']
                    for n in user_notifs.get("task_overdue", {}).itervalues()
                ]
                actual_overdue.sort()
                self.assertEqual(actual_overdue, expected_overdue)

                self.assertEqual([
                    n['title']
                    for n in user_notifs.get("due_today", {}).itervalues()
                ], expected_due_today)

                self.assertEqual([
                    n['title']
                    for n in user_notifs.get("due_in", {}).itervalues()
                ], expected_due_in)
class TestCycleTaskGroupObjectTaskUpdate(TestCase):
    """ This class contains simple cycle_task_group_object_task update tests
  using import functionality
  """

    CSV_DIR = join(abspath(dirname(__file__)), "test_csvs/")

    def setUp(self):
        TestCase.setUp(self)
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        _, self.person_1 = self.object_generator.generate_person(
            user_role="Administrator")
        _, self.person_2 = self.object_generator.generate_person(
            user_role="Administrator")
        self._create_test_cases()

    def _cmp_tasks(self, exp_tasks):
        """Compare tasks values from argument's list and test DB."""
        for exp_slug, exp_task in exp_tasks.iteritems():
            res_task = db.session.query(CycleTaskGroupObjectTask).filter_by(
                slug=exp_slug).first()
            for attr, val in exp_task.iteritems():
                self.assertEqual(str(getattr(res_task, attr, None)), val)

    def test_cycle_task_group_object_task_update(self):
        """Test cycle task group object task update via import"""
        with freeze_time("2016-10-02"):
            # Generate Workflow, Task Groups and Cycle Task Group Object Tasks
            # objects
            _, workflow = self.wf_generator.generate_workflow(
                self.test_workflow)
            self.wf_generator.activate_workflow(workflow)
            start_recurring_cycles()
            # First test: update 4 tasks with correct CSV import
            self._cmp_tasks(self.exp_tasks_before_update)
            filename = "cycle_task_group_object_task_active_update.csv"
            response = self.import_file(filename)
            expected_errors = {
                "Cycle Task Group Object Task": {
                    "block_warnings": {
                        errors.NON_IMPORTABLE_COLUMN_WARNING.format(
                            line=2,
                            column_name='state',
                        ),
                    }
                }
            }
            self._check_csv_response(response, expected_errors)
            self._cmp_tasks(self.exp_tasks_after_update)

    def _create_test_cases(self):
        """Create test cases data"""
        def person_dict(person_id):
            """Return person data"""
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.exp_tasks_before_update = {
            "CYCLETASK-1": {
                "title": "task 1 in tg 1",
                "description": "descr task 1 in tg 1",
                "start_date": "2016-09-30",
                "end_date": "2016-10-07",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": "task 2 in tg 1",
                "description": "descr task 2 in tg 1",
                "start_date": "2016-10-07",
                "end_date": "2016-10-14",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-3": {
                "title": "task 1 in tg 2",
                "description": "descr task 1 in tg 2",
                "start_date": "2016-10-14",
                "end_date": "2016-10-21",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-4": {
                "title": "task 2 in tg 2",
                "description": "descr task 2 in tg 2",
                "start_date": "2016-10-21",
                "end_date": "2016-10-31",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            }
        }

        self.test_workflow = {
            "title":
            "test cycle_task_group_object_task update",
            "notify_on_change":
            False,
            "description":
            "test workflow",
            "owners": [person_dict(self.person_2.id)],
            "frequency":
            "monthly",
            "task_groups": [{
                "title":
                "task group 1",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title":
                    self.exp_tasks_before_update["CYCLETASK-1"]["title"],
                    "description":
                    self.exp_tasks_before_update["CYCLETASK-1"]["description"],
                    "contact":
                    person_dict(self.person_1.id),
                    "relative_start_day":
                    2,
                    "relative_end_day":
                    8,
                }, {
                    "title":
                    self.exp_tasks_before_update["CYCLETASK-2"]["title"],
                    "description":
                    self.exp_tasks_before_update["CYCLETASK-2"]["description"],
                    "contact":
                    person_dict(self.person_1.id),
                    "relative_start_day":
                    9,
                    "relative_end_day":
                    15,
                }],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "another one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title":
                    self.exp_tasks_before_update["CYCLETASK-3"]["title"],
                    "description":
                    self.exp_tasks_before_update["CYCLETASK-3"]["description"],
                    "contact":
                    person_dict(self.person_1.id),
                    "relative_start_day":
                    16,
                    "relative_end_day":
                    22,
                }, {
                    "title":
                    self.exp_tasks_before_update["CYCLETASK-4"]["title"],
                    "description":
                    self.exp_tasks_before_update["CYCLETASK-4"]["description"],
                    "contact":
                    person_dict(self.person_2.id),
                    "relative_start_day":
                    23,
                    "relative_end_day":
                    31,
                }],
                "task_group_objects": []
            }]
        }

        self.exp_tasks_after_update = {
            "CYCLETASK-1": {
                "title": "first task in task group 1",
                "description": "details 1",
                "start_date": "2016-11-01",
                "end_date": "2016-11-05",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": "second task in task group 1",
                "description": "details 2",
                "start_date": "2016-11-06",
                "end_date": "2016-11-10",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-3": {
                "title": "first task in task group 2",
                "description": "details 3",
                "start_date": "2016-11-11",
                "end_date": "2016-11-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-4": {
                "title": "second task in task group 2",
                "description": "details 4",
                "start_date": "2016-11-19",
                "end_date": "2016-11-30",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            }
        }
class TestBacklogWorkflow(TestCase):
  """Test cases for backlog workflow"""

  def setUp(self):  # noqa
    super(TestBacklogWorkflow, self).setUp()
    self.api = Api()
    self.generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.create_backlog_workflow()

  def tearDown(self):
    pass

  def create_backlog_workflow(self):  # pylint: disable=no-self-use
    """Creates one time backlog workflow in database."""

    Workflow.ensure_backlog_workflow_exists()

  def test_add_ctgot_to_backlog_workflow(self):  # pylint: disable=invalid-name
    """Check that backlog workflow exists and test it's basic functionality"""
    # find a backlog workflow
    backlog_workflow = db.session\
                         .query(Workflow)\
                         .filter(Workflow.kind == "Backlog").one()
    backlog_workflow_context_id = backlog_workflow.context.id

    # check that it has an active cycle and a cycle task group
    self.assertEqual(len(backlog_workflow.cycles), 1)
    backlog_cycle = backlog_workflow.cycles[0]
    self.assertEqual(backlog_cycle.is_current, 1)

    # check if it has a cycle task group
    self.assertEqual(len(backlog_cycle.cycle_task_groups), 1)
    backlog_cycle_task_group = backlog_cycle.cycle_task_groups[0]

    # Check that backlog workflow has no workflow people
    self.assertEqual(len(backlog_workflow.people), 0)
    # create a cycle task with creator and put it in backlog workflow
    _, creator = self.object_generator.generate_person(user_role="Creator")
    self.api.set_user(creator)

    # add a task that finishes before the first task in the cycle
    cycle_task_json = {'cycle_task_group_object_task': {
        "title": "Cycle task for backlog",
        "cycle": {"id": backlog_cycle.id, "type": "Cycle"},
        "status": "Assigned",
        "cycle_task_group": {
            "id": backlog_cycle_task_group.id,
            "type": "CycleTaskGroup"
        },
        "start_date": "07/1/2015",
        "end_date": "07/2/2015",
        "task_type": "text",
        "context": {
            "id": backlog_workflow_context_id,
            "type": "Context"
        },
        "task_group_task": {"id": 0, "type": "TaskGroupTask"}
    }}

    response = self.generator.api.post(CycleTaskGroupObjectTask,
                                       cycle_task_json)
    self.assertEqual(response.status_code, 201)

    backlog_cycle_task_group = db.session.query(CycleTaskGroup).filter(
        CycleTaskGroup.cycle_id == backlog_cycle.id).one()
    # Check that changes were not propagated to backlog's CycleTaskGroup
    self.assertEqual(backlog_cycle_task_group.status, "InProgress")
    self.assertEqual(backlog_cycle_task_group.start_date, None)
    self.assertEqual(backlog_cycle_task_group.end_date, None)

    backlog_cycle = db.session.query(Cycle).filter(
        Cycle.id == backlog_cycle.id).one()
    # Check that cycle is still running
    self.assertEqual(backlog_cycle.is_current, 1)
    self.assertEqual(backlog_cycle.status, "Assigned")

  @mock.patch('ggrc_basic_permissions.get_current_user')
  def test_permissions_for_backlog_workflow(self, mock_get_current_user):  # noqa # pylint: disable=invalid-name
    """Tests whether the creator has all the necessary permissions for
    backlog workflow."""

    my_person = Person(name="kekec", email="*****@*****.**")
    creator_role = Role.query.filter(Role.name == "Creator").one()
    user_role = UserRole(role=creator_role, person=my_person)  # noqa # pylint: disable=unused-variable
    mock_get_current_user.return_value = Person(name="Mojca",
                                                email="*****@*****.**")
    backlog_workflow = db.session\
                         .query(Workflow)\
                         .filter(Workflow.kind == "Backlog").one()
    workflow_ctx = backlog_workflow.context.id
    user_perms = load_permissions_for(my_person)

    actions = ["read", "edit", "update"]
    _types = ["Workflow", "Cycle", "CycleTaskGroup",
              "CycleTaskGroupObjectTask", "TaskGroup"]

    for action in actions:
      for obj_type in _types:
        self.assertTrue(workflow_ctx in
                        user_perms[action][obj_type]['contexts'])
    ctgot_ctxs = user_perms['delete']['CycleTaskGroupObjectTask']['contexts']
    self.assertTrue(workflow_ctx in ctgot_ctxs)
class TestMonthlyWorkflowNotification(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        super(TestMonthlyWorkflowNotification, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.person_1 = self.create_user_with_role(role="Administrator")
        self.person_2 = self.create_user_with_role(role="Administrator")
        self.secondary_assignee = self.create_user_with_role(role="Reader")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = dt.datetime.now()

            return new_init

        all_models.Notification.__init__ = init_decorator(
            all_models.Notification.__init__)

    @patch("ggrc.notifications.common.send_email")
    def test_auto_generate_cycle(self, mock_mail):
        """Test auto recurring cycles"""

        with freeze_time("2015-04-01"):
            _, wf = self.wf_generator.generate_workflow(
                self.monthly_workflow_1)
            self.wf_generator.activate_workflow(wf)
            _, notif_data = common.get_daily_notifications()
            contact = self.person_1
            task_assignees = [contact, self.secondary_assignee]

            for user in task_assignees:
                self.assertNotIn(user.email, notif_data)

        with freeze_time("2015-04-02"):
            self.api.client.get("nightly_cron_endpoint")
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertNotIn(user.email, notif_data)
            start_recurring_cycles()
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertNotIn(user.email, notif_data)

        # cycle starts on monday - 6th, and not on 5th
        with freeze_time("2015-04-03"):
            from ggrc.login import noop
            noop.login()
            start_recurring_cycles()
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[contact.email])

        with freeze_time("2015-04-15"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

        with freeze_time("2015-04-25"):  # due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_manual_generate_cycle(self, mock_mail):
        """Test generation of manual cycles"""

        with freeze_time("2015-04-01"):
            _, wf = self.wf_generator.generate_workflow(
                self.monthly_workflow_1)
            self.wf_generator.activate_workflow(wf)

        with freeze_time("2015-04-03"):
            _, cycle = self.wf_generator.generate_cycle(wf)
            contact = self.person_1
            task_assignees = [contact, self.secondary_assignee]
            _, notif_data = common.get_daily_notifications()

            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[contact.email])

        with freeze_time("2015-05-03"):  # two days before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        task_secondary_assignee = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Secondary Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        self.monthly_workflow_1 = {
            "title":
            "test monthly wf notifications",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            # admin will be current user with id == 1
            "unit":
            "month",
            "recurrences":
            True,
            "repeat_every":
            1,
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title":
                    "task 1",
                    "description":
                    "some task",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "contact":
                    person_dict(self.person_1.id),
                    "start_date":
                    dt.date(2015, 4, 5),
                    "end_date":
                    dt.date(2015, 4, 25),
                }, {
                    "title":
                    "task 2",
                    "description":
                    "some task",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "contact":
                    person_dict(self.person_1.id),
                    "start_date":
                    dt.date(2015, 4, 10),
                    "end_date":
                    dt.date(2015, 4, 21),
                }],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "another one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title":
                    "task 1 in tg 2",
                    "description":
                    "some task",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "contact":
                    person_dict(self.person_1.id),
                    "start_date":
                    dt.date(2015, 4, 15),
                    "end_date":
                    dt.date(2015, 4, 15),
                }, {
                    "title":
                    "task 2 in tg 2",
                    "description":
                    "some task",
                    "access_control_list": [
                        acl_helper.get_acl_json(task_secondary_assignee,
                                                self.secondary_assignee.id),
                    ],
                    "contact":
                    person_dict(self.person_2.id),
                    "start_date":
                    dt.date(2015, 4, 15),
                    "end_date":
                    dt.date(2015, 4, 28),
                }],
                "task_group_objects": []
            }]
        }
class TestBacklogWorkflow(TestCase):
  """Test cases for backlog workflow"""

  def setUp(self):  # noqa
    super(TestBacklogWorkflow, self).setUp()
    self.api = Api()
    self.generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.create_backlog_workflow()

  def tearDown(self):
    pass

  def create_backlog_workflow(self):  # pylint: disable=no-self-use
    """Creates one time backlog workflow in database."""

    Workflow.ensure_backlog_workflow_exists()

  def test_add_ctgot_to_backlog_workflow(self):  # pylint: disable=invalid-name
    """Check that backlog workflow exists and test it's basic functionality"""
    # find a backlog workflow
    backlog_workflow = db.session\
                         .query(Workflow)\
                         .filter(and_(Workflow.kind == "Backlog",
                                      Workflow.frequency == "one_time")).one()
    backlog_workflow_context_id = backlog_workflow.context.id

    # check that it has an active cycle and a cycle task group
    self.assertEqual(len(backlog_workflow.cycles), 1)
    backlog_cycle = backlog_workflow.cycles[0]
    self.assertEqual(backlog_cycle.is_current, 1)

    # check if it has a cycle task group
    self.assertEqual(len(backlog_cycle.cycle_task_groups), 1)
    backlog_cycle_task_group = backlog_cycle.cycle_task_groups[0]

    # Check that backlog workflow has no workflow people
    self.assertEqual(len(backlog_workflow.people), 0)
    # create a cycle task with creator and put it in backlog workflow
    _, creator = self.object_generator.generate_person(user_role="Creator")
    self.api.set_user(creator)

    # add a task that finishes before the first task in the cycle
    cycle_task_json = {'cycle_task_group_object_task': {
        "title": "Cycle task for backlog",
        "cycle": {"id": backlog_cycle.id, "type": "Cycle"},
        "status": "Assigned",
        "cycle_task_group": {
            "id": backlog_cycle_task_group.id,
            "type": "CycleTaskGroup"
        },
        "start_date": "07/1/2015",
        "end_date": "07/2/2015",
        "task_type": "text",
        "context": {
            "id": backlog_workflow_context_id,
            "type": "Context"
        },
        "task_group_task": {"id": 0, "type": "TaskGroupTask"}
    }}

    response = self.generator.api.post(CycleTaskGroupObjectTask,
                                       cycle_task_json)
    self.assertEqual(response.status_code, 201)

    backlog_cycle_task_group = db.session.query(CycleTaskGroup).filter(
        CycleTaskGroup.cycle_id == backlog_cycle.id).one()
    # Check that changes were not propagated to backlog's CycleTaskGroup
    self.assertEqual(backlog_cycle_task_group.status, "InProgress")
    self.assertEqual(backlog_cycle_task_group.start_date, None)
    self.assertEqual(backlog_cycle_task_group.end_date, None)

    backlog_cycle = db.session.query(Cycle).filter(
        Cycle.id == backlog_cycle.id).one()
    # Check that cycle is still running
    self.assertEqual(backlog_cycle.is_current, 1)
    self.assertEqual(backlog_cycle.status, "Assigned")

  @mock.patch('ggrc_basic_permissions.get_current_user')
  def test_permissions_for_backlog_workflow(self, mock_get_current_user):  # noqa # pylint: disable=invalid-name
    """Tests whether the creator has all the necessary permissions for
    backlog workflow."""

    my_person = Person(name="kekec", email="*****@*****.**")
    creator_role = Role.query.filter(Role.name == "Creator").one()
    user_role = UserRole(role=creator_role, person=my_person)  # noqa # pylint: disable=unused-variable
    mock_get_current_user.return_value = Person(name="Mojca",
                                                email="*****@*****.**")
    backlog_workflow = db.session\
                         .query(Workflow)\
                         .filter(and_(Workflow.kind == "Backlog",
                                      Workflow.frequency == "one_time")).one()
    workflow_ctx = backlog_workflow.context.id
    user_perms = load_permissions_for(my_person)

    actions = ["read", "edit", "update"]
    _types = ["Workflow", "Cycle", "CycleTaskGroup",
              "CycleTaskGroupObjectTask", "TaskGroup"]

    for action in actions:
      for obj_type in _types:
        self.assertTrue(workflow_ctx in
                        user_perms[action][obj_type]['contexts'])
    ctgot_ctxs = user_perms['delete']['CycleTaskGroupObjectTask']['contexts']
    self.assertTrue(workflow_ctx in ctgot_ctxs)
class TestRecurringWorkflowRevisions(TestCase):
  """Starting start recurring cycle should generate revisions."""

  def setUp(self):
    super(TestRecurringWorkflowRevisions, self).setUp()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    _, self.person_1 = self.object_generator.generate_person(
        user_role="Administrator")
    _, self.person_2 = self.object_generator.generate_person(
        user_role="Administrator")

    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }
    self.monthly_workflow = {
        "title": "test monthly wf notifications",
        "notify_on_change": True,
        "description": "some test workflow",
        "owners": [person_dict(self.person_2.id)],
        "unit": "month",
        "repeat_every": 1,
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.person_1.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
            }, {
                "title": "task 2",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "contact": person_dict(self.person_1.id),
            "task_group_tasks": [{
                "title": "task 1 in tg 2",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
            }, {
                "title": "task 2 in tg 2",
                "description": "some task",
                "contact": person_dict(self.person_2.id),
            }],
            "task_group_objects": []
        }]
    }

  @unittest.skip("Required to fix log event procedure for new calculator")
  @patch("ggrc.notifications.common.send_email")
  def test_revisions(self, mock_mail):  # pylint: disable=unused-argument
    with freeze_time("2015-04-01"):
      _, workflow = self.wf_generator.generate_workflow(self.monthly_workflow)
      self.wf_generator.activate_workflow(workflow)
    event_count = Event.query.count()
    revision_query = Revision.query.filter_by(
        resource_type='CycleTaskGroupObjectTask',
    )
    revision_count = revision_query.count()
    # cycle starts on monday - 6th, and not on 5th
    with freeze_time("2015-04-03"):
      start_recurring_cycles()

    self.assertEqual(event_count + 1, Event.query.count())
    self.assertNotEqual(revision_count, revision_query.count())
class TestOneTimeWorkflowNotification(TestCase):

  """ Tests are defined in the g-sheet test grid under:
    WF EMAILS for unit tests (middle level)
  """

  def setUp(self):
    super(TestOneTimeWorkflowNotification, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.random_people = [
        self.object_generator.generate_person(user_role="Administrator")[1]
        for _ in range(5)]
    self.create_test_cases()

    self.create_users()

    db.session.query(Notification).delete()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  def tearDown(self):
    db.session.query(Notification).delete()

  def short_dict(self, obj, plural):
    return {
        "href": "/api/%s/%d" % (plural, obj.id),
        "id": obj.id,
        "type": obj.__class__.__name__,
    }

  def test_one_time_wf(self):
    # setup
    with freeze_time("2015-04-07 03:21:34"):
      wf_response, wf = self.wf_generator.generate_workflow(data={
          # admin will be the current user
          "notify_on_change": True,  # force real time updates
          "title": "One-time WF",
          "notify_custom_message": textwrap.dedent("""\
              Hi all.
              Did you know that Irelnd city namd Newtownmountkennedy has 19
              letters? But it's not the longest one. The recordsman is the
              city in New Zealand that contains 97 letter."""),
      })

      _, tg = self.wf_generator.generate_task_group(wf, data={
          "title": "TG #1 for the One-time WF",
          "contact": self.short_dict(self.tgassignee1, "people"),
      })

      self.wf_generator.generate_task_group_task(tg, {
          "title": "task #1 for one-time workflow",
          "contact": self.short_dict(self.member1, "people"),
          "start_date": "04/07/2015",
          "end_date": "04/15/2015",
      })

      self.wf_generator.generate_task_group_object(tg, self.random_objects[0])
      self.wf_generator.generate_task_group_object(tg, self.random_objects[1])

    # test
    with freeze_time("2015-04-07 03:21:34"):
      cycle_response, cycle = self.wf_generator.generate_cycle(wf)
      self.wf_generator.activate_workflow(wf)

      common.get_daily_notifications()

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.one_time_workflow_1 = {
        "title": "one time test workflow",
        "description": "some test workflow",
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "contact": person_dict(self.random_people[0].id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task 2",
                "description": "some task",
                "contact": person_dict(self.random_people[1].id),
                "start_date": date(2015, 5, 4),
                "end_date": date(2015, 5, 7),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "task_group_tasks": [{
                "title": "task 1 in tg 2",
                "description": "some task",
                "contact": person_dict(self.random_people[0].id),
                "start_date": date(2015, 5, 8),  # friday
                "end_date": date(2015, 5, 12),
            }, {
                "title": "task 2 in tg 2",
                "description": "some task",
                "contact": person_dict(self.random_people[2].id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": []
        }]
    }

  def create_users(self):
    _, self.admin1 = self.object_generator.generate_person(
        # data={"name": "User1 Admin1", "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.tgassignee1 = self.object_generator.generate_person(
        # data={"name": "User2 TGassignee1",
        #       "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.member1 = self.object_generator.generate_person(
        # data={"name": "User3 Member1", "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.member2 = self.object_generator.generate_person(
        # data={"name": "User4 Member2", "email": "*****@*****.**"},
        user_role="Administrator")
class TestOneTimeWfEndDateChange(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        super(TestOneTimeWfEndDateChange, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        all_models.Notification.query.delete()

        self.random_objects = self.object_generator.generate_random_objects(2)
        self.user = self.create_user_with_role(role="Administrator")
        self.secondary_assignee = self.create_user_with_role(role="Reader")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        all_models.Notification.__init__ = init_decorator(
            all_models.Notification.__init__)

    @patch("ggrc.notifications.common.send_email")
    def test_no_date_change(self, mock_mail):
        """Test a basic case with no moving end date"""
        with freeze_time("2015-04-10 03:21:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        assignee = self.user
        task_assignees = [assignee, self.secondary_assignee]
        with freeze_time("2015-04-11 03:21:34"):
            _, notif_data = common.get_daily_notifications()

            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[assignee.email])

        with freeze_time("2015-05-02 03:21:34"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(assignee.email, notif_data)

            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[assignee.email])
            for user in task_assignees:
                self.assertNotIn("due_in", notif_data[user.email])
                self.assertNotIn("due_today", notif_data[user.email])

        with freeze_time("2015-05-02 03:21:34"):
            common.send_daily_digest_notifications()
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

            # one email to admin, one to assigne and one to secondary assignee
            self.assertEqual(mock_mail.call_count, 3)

        with freeze_time("2015-05-04 03:21:34"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertIn("due_in", notif_data[user.email])
                self.assertEqual(len(notif_data[user.email]["due_in"]), 2)

        with freeze_time("2015-05-04 03:21:34"):  # one day before due date
            common.send_daily_digest_notifications()
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

            # one email to admin and two each to assigne and secondary assignee
            self.assertEqual(mock_mail.call_count, 5)

        with freeze_time("2015-05-05 03:21:34"):  # due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn("due_today", notif_data[user.email])
                self.assertEqual(len(notif_data[user.email]["due_today"]), 2)

    @patch("ggrc.notifications.common.send_email")
    def test_move_end_date_to_future(self, mock_mail):
        """Test moving the end date to the future.

    It is done before due_in and due_today notifications have been sent.
    """
        with freeze_time("2015-04-10 03:21:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        assignee = self.user
        task_assignees = [assignee, self.secondary_assignee]
        with freeze_time("2015-04-11 03:21:34"):
            _, notif_data = common.get_daily_notifications()
            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[assignee.email])

        with freeze_time("2015-05-02 03:21:34"):
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertNotIn("due_in", notif_data[user.email])
                self.assertNotIn("due_today", notif_data[user.email])
            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[assignee.email])

        with freeze_time("2015-05-02 03:21:34"):
            common.send_daily_digest_notifications()
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

            # one email to admin and one each to assigne and secondary assignee
            self.assertEqual(mock_mail.call_count, 3)

        with freeze_time("2015-05-03 03:21:34"):
            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)
            task2 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[1].id)

            self.wf_generator.modify_object(
                task1, data={"end_date": date(2015, 5, 15)})
            self.wf_generator.modify_object(
                task2, data={"end_date": date(2015, 5, 15)})

        with freeze_time("2015-05-04 03:21:34"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        with freeze_time("2015-05-05 03:21:34"):  # due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        with freeze_time("2015-05-14 03:21:34"):  # due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertIn("due_in", notif_data[user.email])
                self.assertEqual(len(notif_data[user.email]["due_in"]),
                                 len(self.random_objects))

        with freeze_time("2015-05-15 03:21:34"):  # due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

                self.assertIn("due_in", notif_data[user.email])

                self.assertIn("due_today", notif_data[user.email])
                self.assertEqual(len(notif_data[user.email]["due_today"]),
                                 len(self.random_objects))

    @patch("ggrc.notifications.common.send_email")
    def test_move_end_date_to_past(self, mock_mail):
        """Test moving an end date to the past"""
        with freeze_time("2015-04-10 03:21:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-02 03:21:34"):
            common.send_daily_digest_notifications()
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

            # one email to admin and one to assignee
            self.assertEqual(mock_mail.call_count, 3)

        with freeze_time("2015-05-03 03:21:34"):
            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)
            task2 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[1].id)

            self.wf_generator.modify_object(
                task1, data={"end_date": date(2015, 5, 1)})
            self.wf_generator.modify_object(
                task2, data={"end_date": date(2015, 5, 1)})

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time("2015-05-03 03:21:34"):  # two days after due date
            _, notif_data = common.get_daily_notifications()
            self.assertNotEqual(notif_data, {})
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

                user_notifs = notif_data[user.email]
                self.assertNotIn("due_today", user_notifs)
                self.assertNotIn("due_in", user_notifs)

                self.assertIn("task_overdue", user_notifs)
                self.assertEqual(len(user_notifs["task_overdue"]), 2)

    @patch("ggrc.notifications.common.send_email")
    def test_move_end_date_to_today(self, mock_mail):
        """Test moving end date to today"""
        with freeze_time("2015-04-10 03:21:34"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        task_assignees = [self.user, self.secondary_assignee]
        with freeze_time("2015-05-02 03:21:34"):
            common.send_daily_digest_notifications()
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

            # one email to admin and one to assigne
            self.assertEqual(mock_mail.call_count, 3)

        with freeze_time("2015-05-03 03:21:34"):
            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)
            task2 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[1].id)

            self.wf_generator.modify_object(
                task1, data={"end_date": date(2015, 5, 3)})
            self.wf_generator.modify_object(
                task2, data={"end_date": date(2015, 5, 4)})

        with freeze_time("2015-05-03 03:21:34"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertNotEqual(notif_data, {})
                self.assertIn(user.email, notif_data)
                self.assertIn("due_today", notif_data[user.email])
                self.assertIn("due_in", notif_data[user.email])
                self.assertEqual(len(notif_data[user.email]["due_today"]), 1)

            common.send_daily_digest_notifications()

        with freeze_time("2015-05-04 03:21:34"):  # due date (of task2)
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertIn("due_today", notif_data[user.email])
                self.assertNotIn("due_in", notif_data[user.email])
            common.send_daily_digest_notifications()

        # check that overdue notifications are sent even on days after the day a
        # task has become due, unlike the "due_today" and "due_in" notifications
        with freeze_time("2015-05-05 03:21:34"):  # 1+ days after due date(s)
            _, notif_data = common.get_daily_notifications()
            self.assertNotEqual(notif_data, {})
            for user in task_assignees:
                self.assertIn(user.email, notif_data)

                user_notifs = notif_data[user.email]
                self.assertNotIn("due_today", user_notifs)
                self.assertNotIn("due_in", user_notifs)

                self.assertIn("task_overdue", user_notifs)
                self.assertEqual(len(user_notifs["task_overdue"]), 2)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        task_assignee_role_id = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        task_secondary_assignee = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Secondary Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            # admin will be current user with id == 1
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.user.id),
                "task_group_tasks": [
                    {
                        "title":
                        "task 1",
                        "description":
                        "some task",
                        "start_date":
                        date(2015, 5, 1),  # friday
                        "end_date":
                        date(2015, 5, 5),
                        "access_control_list": [
                            acl_helper.get_acl_json(task_assignee_role_id,
                                                    self.user.id),
                            acl_helper.get_acl_json(
                                task_secondary_assignee,
                                self.secondary_assignee.id),
                        ],
                    },
                    {
                        "title":
                        "task 2",
                        "description":
                        "some task 2",
                        "start_date":
                        date(2015, 5, 1),  # friday
                        "end_date":
                        date(2015, 5, 5),
                        "access_control_list": [
                            acl_helper.get_acl_json(task_assignee_role_id,
                                                    self.user.id),
                            acl_helper.get_acl_json(
                                task_secondary_assignee,
                                self.secondary_assignee.id),
                        ],
                    }
                ],
                "task_group_objects":
                self.random_objects
            }]
        }
class TestCycleTaskStatusChange(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestCycleTaskStatusChange, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  def test_task_declined_notification_created(self):
    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1, "Declined")

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == task1.id,
          Notification.object_type == task1.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type == self.get_notification_type(
              "cycle_task_declined"
          )
      )).all()

      self.assertEqual(len(notif), 1, "notifications: {}".format(str(notif)))

  def test_all_tasks_finished_notification_created(self):
    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1)

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == cycle.id,
          Notification.object_type == cycle.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type == self.get_notification_type(
              "all_cycle_tasks_completed"
          )
      )).all()

      self.assertEqual(len(notif), 1, "notifications: {}".format(str(notif)))

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == task1.id,
          Notification.object_type == task1.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type != self.get_notification_type(
              "all_cycle_tasks_completed"
          )
      )).all()

      self.assertEqual(notif, [])

  def test_multi_all_tasks_finished_notification_created(self):

    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_2)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1)

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == cycle.id,
          Notification.object_type == cycle.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type == self.get_notification_type(
              "all_cycle_tasks_completed"
          )
      )).all()

      # there is still one task in the cycle, so there should be no
      # notifications for all tasks completed
      self.assertEqual(notif, [])

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == task1.id,
          Notification.object_type == task1.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type != self.get_notification_type(
              "all_cycle_tasks_completed"
          )
      )).all()

      # The task was verified, so there should be no notifications left for due
      # dates.
      self.assertEqual(notif, [])

      task2 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[1].id)

      self.task_change_status(task2)

      notif = db.session.query(Notification).filter(and_(
          Notification.object_id == cycle.id,
          Notification.object_type == cycle.type,
          Notification.sent_at == None,  # noqa
          Notification.notification_type == self.get_notification_type(
              "all_cycle_tasks_completed"
          )
      )).all()

      self.assertEqual(len(notif), 1, "notifications: {}".format(str(notif)))

  @patch("ggrc.notifications.common.send_email")
  def test_single_task_declined(self, mock_mail):
    """
    test moving the end date to the future, befor due_in and due_today
    notifications have been sent
    """

    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-05-02"):
      common.send_daily_digest_notifications()

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1, "Finished")

      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-02"):
      common.send_daily_digest_notifications()

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1, "Declined")

      user = Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()

      self.assertIn(user.email, notif_data)
      self.assertIn("task_declined", notif_data[user.email])

  @patch("ggrc.notifications.common.send_email")
  def test_single_task_accepted(self, mock_mail):
    """
    test moving the end date to the future, befor due_in and due_today
    notifications have been sent
    """

    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-05-02"):
      common.send_daily_digest_notifications()

      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1, "Finished")

      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-03"):
      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)

      self.task_change_status(task1)

      user = Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)
      self.assertIn("all_tasks_completed", notif_data["*****@*****.**"])

  @patch("ggrc.notifications.common.send_email")
  def test_end_cycle(self, mock_mail):
    """
    manaually ending a cycle should stop all notifications for that cycle
    """

    with freeze_time("2015-05-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)
      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-05-03"):
      _, notif_data = common.get_daily_notifications()
      cycle = Cycle.query.get(cycle.id)
      user = Person.query.get(self.user.id)
      self.assertIn(user.email, notif_data)
      self.wf_generator.modify_object(cycle, data={"is_current": False})
      cycle = Cycle.query.get(cycle.id)
      self.assertFalse(cycle.is_current)

      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.one_time_workflow_1 = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "owners": [person_dict(self.user.id)],
        "task_groups": [{
            "title": "single task group",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "single task in a wf",
                "contact": person_dict(self.user.id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
        }]
    }

    self.one_time_workflow_2 = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "owners": [person_dict(self.user.id)],
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "two taks in wf with different objects",
                "contact": person_dict(self.user.id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task 2",
                "description": "two taks in wf with different objects",
                "contact": person_dict(self.user.id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": self.random_objects
        }]
    }

  def get_notification_type(self, name):
    return db.session.query(NotificationType).filter(
        NotificationType.name == name).one()

  def task_change_status(self, task, status="Verified"):
    self.wf_generator.modify_object(
        task, data={"status": status})

    task = CycleTaskGroupObjectTask.query.get(task.id)

    self.assertEqual(task.status, status)
class TestNotificationsForDeletedObjects(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    TestCase.setUp(self)
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_delete_activated_workflow(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, workflow = self.wf_generator.generate_workflow(self.quarterly_wf_1)
      response, workflow = self.wf_generator.activate_workflow(workflow)

      self.assert200(response)

      user = Person.query.get(self.user.id)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_starts_in", notif_data[user.email])

      workflow = Workflow.query.get(workflow.id)

      response = self.wf_generator.api.delete(workflow)
      self.assert200(response)

      _, notif_data = common.get_daily_notifications()
      user = Person.query.get(self.user.id)

      self.assertNotIn(user.email, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.quarterly_wf_1 = {
        "title": "quarterly wf 1",
        "notify_on_change": True,
        "description": "",
        "owners": [person_dict(self.user.id)],
        "frequency": "quarterly",
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
                "relative_start_day": 5,
                "relative_start_month": 2,
                "relative_end_day": 25,
                "relative_end_month": 2,
            },
            ],
        },
        ]
    }
Exemple #18
0
class TestMonthlyWorkflowNotification(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestMonthlyWorkflowNotification, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    _, self.person_1 = self.object_generator.generate_person(
        user_role="Administrator")
    _, self.person_2 = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = dt.datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_auto_generate_cycle(self, mock_mail):

    person_1_email = Person.query.get(self.person_1.id).email
    with freeze_time("2015-04-01"):
      _, wf = self.wf_generator.generate_workflow(self.monthly_workflow_1)
      self.wf_generator.activate_workflow(wf)
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(person_1_email, notif_data)

    with freeze_time("2015-04-02"):
      self.api.client.get("nightly_cron_endpoint")
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(person_1_email, notif_data)
      start_recurring_cycles()
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(person_1_email, notif_data)

    # cycle starts on monday - 6th, and not on 5th
    with freeze_time("2015-04-03"):
      start_recurring_cycles()
      _, notif_data = common.get_daily_notifications()
      self.assertIn(person_1_email, notif_data)
      self.assertIn("cycle_started", notif_data[person_1_email])

    with freeze_time("2015-04-15"):  # one day befor due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(person_1_email, notif_data)

    with freeze_time("2015-04-25"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(person_1_email, notif_data)

  @patch("ggrc.notifications.common.send_email")
  def test_manual_generate_cycle(self, mock_mail):

    with freeze_time("2015-04-01"):
      _, wf = self.wf_generator.generate_workflow(self.monthly_workflow_1)
      self.wf_generator.activate_workflow(wf)

      person_1 = Person.query.get(self.person_1.id)

    with freeze_time("2015-04-03"):
      _, notif_data = common.get_daily_notifications()

    with freeze_time("2015-04-03"):
      _, cycle = self.wf_generator.generate_cycle(wf)
      _, notif_data = common.get_daily_notifications()
      person_1 = Person.query.get(self.person_1.id)
      self.assertIn("cycle_started", notif_data[person_1.email])

    with freeze_time("2015-05-03"):  # two days befor due date
      _, notif_data = common.get_daily_notifications()
      person_1 = Person.query.get(self.person_1.id)
      self.assertIn(person_1.email, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.monthly_workflow_1 = {
        "title": "test monthly wf notifications",
        "notify_on_change": True,
        "description": "some test workflow",
        # admin will be current user with id == 1
        "unit": "month",
        "recurrences": True,
        "repeat_every": 1,
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.person_1.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
                "start_date": dt.date(2015, 4, 5),
                "end_date": dt.date(2015, 4, 25),
            }, {
                "title": "task 2",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
                "start_date": dt.date(2015, 4, 10),
                "end_date": dt.date(2015, 4, 21),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "contact": person_dict(self.person_1.id),
            "task_group_tasks": [{
                "title": "task 1 in tg 2",
                "description": "some task",
                "contact": person_dict(self.person_1.id),
                "start_date": dt.date(2015, 4, 15),
                "end_date": dt.date(2015, 4, 15),
            }, {
                "title": "task 2 in tg 2",
                "description": "some task",
                "contact": person_dict(self.person_2.id),
                "start_date": dt.date(2015, 4, 15),
                "end_date": dt.date(2015, 4, 28),
            }],
            "task_group_objects": []
        }]
    }
Exemple #19
0
class TestOneTimeWfEndDateChange(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestOneTimeWfEndDateChange, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_no_date_change(self, mock_mail):
    def get_person(person_id):
      return db.session.query(Person).filter(Person.id == person_id).one()

    with freeze_time("2015-04-10 03:21:34"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-04-11 03:21:34"):
      user = get_person(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn("cycle_started", notif_data[user.email])

    with freeze_time("2015-05-02 03:21:34"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_started", notif_data[user.email])
      self.assertNotIn("due_in", notif_data[user.email])
      self.assertNotIn("due_today", notif_data[user.email])

    with freeze_time("2015-05-02 03:21:34"):
      common.send_daily_digest_notifications()
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

      # one email to owner and one to assigne
      self.assertEqual(mock_mail.call_count, 2)

    with freeze_time("2015-05-04 03:21:34"):  # one day before due date
      _, notif_data = common.get_daily_notifications()
      user = get_person(self.user.id)
      self.assertIn("due_in", notif_data[user.email])
      self.assertEqual(len(notif_data[user.email]["due_in"]), 2)

    with freeze_time("2015-05-04 03:21:34"):  # one day before due date
      common.send_daily_digest_notifications()
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

      # one email to owner and one to assigne
      self.assertEqual(mock_mail.call_count, 3)

    with freeze_time("2015-05-05 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn("due_today", notif_data[user.email])
      self.assertEqual(len(notif_data[user.email]["due_today"]), 2)

  @patch("ggrc.notifications.common.send_email")
  def test_move_end_date_to_future(self, mock_mail):
    """
    test moving the end date to the future, befor due_in and due_today
    notifications have been sent
    """
    def get_person(person_id):
      return db.session.query(Person).filter(Person.id == person_id).one()

    with freeze_time("2015-04-10 03:21:34"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-04-11 03:21:34"):
      user = get_person(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn("cycle_started", notif_data[user.email])

    with freeze_time("2015-05-02 03:21:34"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_started", notif_data[user.email])
      self.assertNotIn("due_in", notif_data[user.email])
      self.assertNotIn("due_today", notif_data[user.email])

    with freeze_time("2015-05-02 03:21:34"):
      common.send_daily_digest_notifications()
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

      # one email to owner and one to assigne
      self.assertEqual(mock_mail.call_count, 2)

    with freeze_time("2015-05-03 03:21:34"):
      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)
      task2 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[1].id)

      self.wf_generator.modify_object(
          task1, data={"end_date": date(2015, 5, 15)})
      self.wf_generator.modify_object(
          task2, data={"end_date": date(2015, 5, 15)})

    with freeze_time("2015-05-04 03:21:34"):  # one day befor due date
      _, notif_data = common.get_daily_notifications()
      user = get_person(self.user.id)
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-05 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-14 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("due_in", notif_data[user.email])
      self.assertEqual(len(notif_data[user.email]["due_in"]),
                       len(self.random_objects))

    with freeze_time("2015-05-15 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

      # yesterdays mail has not been sent
      self.assertIn("due_in", notif_data[user.email])

      self.assertIn("due_today", notif_data[user.email])
      self.assertEqual(len(notif_data[user.email]["due_today"]),
                       len(self.random_objects))

  @patch("ggrc.notifications.common.send_email")
  def test_move_end_date_to_past(self, mock_mail):
    def get_person(person_id):
      return db.session.query(Person).filter(Person.id == person_id).one()

    with freeze_time("2015-04-10 03:21:34"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-05-02 03:21:34"):
      common.send_daily_digest_notifications()
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

      # one email to owner and one to assigne
      self.assertEqual(mock_mail.call_count, 2)

    with freeze_time("2015-05-03 03:21:34"):
      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)
      task2 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[1].id)

      self.wf_generator.modify_object(
          task1, data={"end_date": date(2015, 5, 1)})
      self.wf_generator.modify_object(
          task2, data={"end_date": date(2015, 5, 1)})

    with freeze_time("2015-05-03 03:21:34"):  # one day befor due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-04 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

    with freeze_time("2015-05-05 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

  @patch("ggrc.notifications.common.send_email")
  def test_move_end_date_to_today(self, mock_mail):
    def get_person(person_id):
      return db.session.query(Person).filter(Person.id == person_id).one()

    with freeze_time("2015-04-10 03:21:34"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)

    with freeze_time("2015-05-02 03:21:34"):
      common.send_daily_digest_notifications()
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

      # one email to owner and one to assigne
      self.assertEqual(mock_mail.call_count, 2)

    with freeze_time("2015-05-03 03:21:34"):
      cycle = Cycle.query.get(cycle.id)
      task1 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[0].id)
      task2 = CycleTaskGroupObjectTask.query.get(
          cycle.cycle_task_group_object_tasks[1].id)

      self.wf_generator.modify_object(
          task1, data={"end_date": date(2015, 5, 3)})
      self.wf_generator.modify_object(
          task2, data={"end_date": date(2015, 5, 4)})

    with freeze_time("2015-05-03 03:21:34"):  # one day befor due date
      user = get_person(self.user.id)
      _, notif_data = common.get_daily_notifications()

      self.assertNotEquals(notif_data, {})
      self.assertIn(user.email, notif_data)
      self.assertIn("due_today", notif_data[user.email])
      self.assertIn("due_in", notif_data[user.email])
      self.assertEqual(len(notif_data[user.email]["due_today"]), 1)

      common.send_daily_digest_notifications()

    with freeze_time("2015-05-04 03:21:34"):  # due date
      user = get_person(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("due_today", notif_data[user.email])
      self.assertNotIn("due_in", notif_data[user.email])
      common.send_daily_digest_notifications()

    with freeze_time("2015-05-05 03:21:34"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(notif_data, {})

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.one_time_workflow_1 = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "owners": [person_dict(self.user.id)],
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "contact": person_dict(self.user.id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task 2",
                "description": "some task 2",
                "contact": person_dict(self.user.id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": self.random_objects
        }]
    }
class TestCycleTaskStatusChange(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        TestCase.setUp(self)
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        Notification.query.delete()

        self.random_objects = self.object_generator.generate_random_objects(2)
        _, self.user = self.object_generator.generate_person(
            user_role="Administrator")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    def test_task_declined_notification_created(self):
        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Declined")

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == task1.id,
                    Notification.object_type == task1.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type ==
                    self.get_notification_type("cycle_task_declined"))).all()

            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

    def test_all_tasks_finished_notification_created(self):
        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == cycle.id,
                    Notification.object_type == cycle.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type == self.
                    get_notification_type("all_cycle_tasks_completed"))).all()

            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == task1.id,
                    Notification.object_type == task1.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type != self.
                    get_notification_type("all_cycle_tasks_completed"))).all()

            self.assertEqual(notif, [])

    def test_multi_all_tasks_finished_notification_created(self):

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_2)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == cycle.id,
                    Notification.object_type == cycle.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type == self.
                    get_notification_type("all_cycle_tasks_completed"))).all()

            # there is still one task in the cycle, so there should be no
            # notifications for all tasks completed
            self.assertEqual(notif, [])

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == task1.id,
                    Notification.object_type == task1.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type != self.
                    get_notification_type("all_cycle_tasks_completed"))).all()

            # The task was verified, so there should be no notifications left for due
            # dates.
            self.assertEqual(notif, [])

            task2 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[1].id)

            self.task_change_status(task2)

            notif = db.session.query(Notification).filter(
                and_(
                    Notification.object_id == cycle.id,
                    Notification.object_type == cycle.type,
                    Notification.sent_at == None,  # noqa
                    Notification.notification_type == self.
                    get_notification_type("all_cycle_tasks_completed"))).all()

            self.assertEqual(len(notif), 1,
                             "notifications: {}".format(str(notif)))

    @patch("ggrc.notifications.common.send_email")
    def test_single_task_declined(self, mock_mail):
        """
    test moving the end date to the future, befor due_in and due_today
    notifications have been sent
    """

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Finished")

            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Declined")

            user = Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()

            self.assertIn(user.email, notif_data)
            self.assertIn("task_declined", notif_data[user.email])

    @patch("ggrc.notifications.common.send_email")
    def test_single_task_accepted(self, mock_mail):
        """
    test moving the end date to the future, befor due_in and due_today
    notifications have been sent
    """

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-02"):
            common.send_daily_digest_notifications()

            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1, "Finished")

            _, notif_data = common.get_daily_notifications()
            self.assertEqual(notif_data, {})

        with freeze_time("2015-05-03"):
            cycle = Cycle.query.get(cycle.id)
            task1 = CycleTaskGroupObjectTask.query.get(
                cycle.cycle_task_group_object_tasks[0].id)

            self.task_change_status(task1)

            user = Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(user.email, notif_data)
            self.assertIn("all_tasks_completed",
                          notif_data["*****@*****.**"])

    @patch("ggrc.notifications.common.send_email")
    def test_end_cycle(self, mock_mail):
        """
    manaually ending a cycle should stop all notifications for that cycle
    """

        with freeze_time("2015-05-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)
            _, cycle = self.wf_generator.generate_cycle(workflow)
            self.wf_generator.activate_workflow(workflow)

        with freeze_time("2015-05-03"):
            _, notif_data = common.get_daily_notifications()
            cycle = Cycle.query.get(cycle.id)
            user = Person.query.get(self.user.id)
            self.assertIn(user.email, notif_data)
            self.wf_generator.modify_object(cycle, data={"is_current": False})
            cycle = Cycle.query.get(cycle.id)
            self.assertFalse(cycle.is_current)

            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(user.email, notif_data)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "owners": [person_dict(self.user.id)],
            "task_groups": [{
                "title":
                "single task group",
                "contact":
                person_dict(self.user.id),
                "task_group_tasks": [{
                    "title": "task 1",
                    "description": "single task in a wf",
                    "contact": person_dict(self.user.id),
                    "start_date": date(2015, 5, 1),  # friday
                    "end_date": date(2015, 5, 5),
                }],
            }]
        }

        self.one_time_workflow_2 = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "owners": [person_dict(self.user.id)],
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.user.id),
                "task_group_tasks": [
                    {
                        "title": "task 1",
                        "description": "two taks in wf with different objects",
                        "contact": person_dict(self.user.id),
                        "start_date": date(2015, 5, 1),  # friday
                        "end_date": date(2015, 5, 5),
                    },
                    {
                        "title": "task 2",
                        "description": "two taks in wf with different objects",
                        "contact": person_dict(self.user.id),
                        "start_date": date(2015, 5, 1),  # friday
                        "end_date": date(2015, 5, 5),
                    }
                ],
                "task_group_objects":
                self.random_objects
            }]
        }

    def get_notification_type(self, name):
        return db.session.query(NotificationType).filter(
            NotificationType.name == name).one()

    def task_change_status(self, task, status="Verified"):
        self.wf_generator.modify_object(task, data={"status": status})

        task = CycleTaskGroupObjectTask.query.get(task.id)

        self.assertEqual(task.status, status)
Exemple #21
0
class TestCycleStartFailed(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestCycleStartFailed, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_start_failed(self, mock_mail):

    wf_admin = "*****@*****.**"

    with freeze_time("2015-02-01 13:39:20"):
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
      response, wf = self.wf_generator.activate_workflow(wf)
      print wf.next_cycle_start_date

      self.assert200(response)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(wf_admin, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(wf_admin, notif_data)
      self.assertIn("cycle_starts_in", notif_data[wf_admin])

    with freeze_time("2015-03-05 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(wf_admin, notif_data)
      self.assertNotIn("cycle_started", notif_data[wf_admin])
      self.assertIn(wf_admin, notif_data)
      self.assertIn("cycle_start_failed", notif_data[wf_admin])

      common.send_daily_digest_notifications()

      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(wf_admin, notif_data)

  # TODO: investigate why next_cycle_start date remains the same after
  # start_recurring_cycles

  # @patch("ggrc.notifications.common.send_email")
  # def test_start_failed_send_notifications(self, mock_mail):

  #   wf_owner = "*****@*****.**"

  #   with freeze_time("2015-02-01 13:39:20"):
  #     _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
  #     response, wf = self.wf_generator.activate_workflow(wf)
  #     print wf.next_cycle_start_date

  #     self.assert200(response)

  #   with freeze_time("2015-01-01 13:39:20"):
  #     _, notif_data = common.get_daily_notifications()
  #     self.assertNotIn(wf_owner, notif_data)

  #   with freeze_time("2015-01-29 13:39:20"):
  #     _, notif_data = common.get_daily_notifications()
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertIn("cycle_starts_in", notif_data[wf_owner])

  #   with freeze_time("2015-02-05 13:39:20"):
  #     _, notif_data = common.get_daily_notifications()
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertNotIn("cycle_started", notif_data[wf_owner])
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertIn("cycle_start_failed", notif_data[wf_owner])

  #     start_recurring_cycles()
  #     _, notif_data = common.get_daily_notifications()
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertIn("cycle_started", notif_data[wf_owner])
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertNotIn("cycle_start_failed", notif_data[wf_owner])

  #     common.send_daily_digest_notifications()

  #     _, notif_data = common.get_daily_notifications()
  #     self.assertNotIn(wf_owner, notif_data)

  # @patch("ggrc.notifications.common.send_email")
  # def test_start_failed_send_notifications_monthly(self, mock_mail):

  #   wf_owner = "*****@*****.**"

  #   with freeze_time("2015-05-12 13:39:20"):
  #     _, wf = self.wf_generator.generate_workflow(self.monthly)
  #     response, wf = self.wf_generator.activate_workflow(wf)

  #   with freeze_time("2015-05-14 13:39:20"):
  #     _, wf = self.wf_generator.generate_workflow(self.monthly)
  #     response, wf = self.wf_generator.activate_workflow(wf)

  #     self.assert200(response)

  #     _, notif_data = common.get_daily_notifications()
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertNotIn("cycle_started", notif_data[wf_owner])
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertIn("cycle_start_failed", notif_data[wf_owner])

  #     start_recurring_cycles()
  #     _, notif_data = common.get_daily_notifications()
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertIn("cycle_started", notif_data[wf_owner])
  #     self.assertIn(wf_owner, notif_data)
  #     self.assertNotIn("cycle_start_failed", notif_data[wf_owner])

  #     common.send_daily_digest_notifications()

  #     _, notif_data = common.get_daily_notifications()
  #     self.assertNotIn(wf_owner, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.quarterly_wf = {
        "title": "quarterly wf forced notifications",
        "notify_on_change": True,
        "description": "",
        # admin will be current user with id == 1
        "unit": "month",
        "repeat_every": 3,
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
            },
            ],
        },
        ]
    }
    self.monthly = {
        "title": "monthly",
        "notify_on_change": True,
        "description": "",
        # admin will be current user with id == 1
        "unit": "month",
        "repeat_every": 1,
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
                "relative_start_day": 14,
                "relative_end_day": 25,
            },
            ],
        },
        ]
    }
class TestCycleTaskImportUpdate(BaseTestCycleTaskImportUpdate):
    """ This class contains simple cycle task update tests using import
  functionality
  """

    CSV_DIR = join(abspath(dirname(__file__)), "test_csvs/")

    IMPORTABLE_COLUMN_NAMES = [
        "Summary",
        "Task Description",
        "Start Date",
        "Due Date",
        "State",
        "Task Assignees",
        "Task Secondary Assignees",
    ]

    def setUp(self):
        super(TestCycleTaskImportUpdate, self).setUp()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        self.random_objects = self.object_generator.generate_random_objects(2)
        self.person_1 = ggrc_factories.PersonFactory()
        self.ftime_active = "2016-07-01"
        self.ftime_historical = "2014-05-01"
        self._create_test_cases_data()
        # It is needed because cycle-tasks are generated automatically with
        # 'slug' based on auto_increment 'id' field.
        # At start of each test we suppose that created cycle-task's 'slug'
        # lie in range from 1 to 10.
        db.session.execute('ALTER TABLE cycle_task_group_object_tasks '
                           'AUTO_INCREMENT = 1')

    def test_cycle_task_correct(self):
        """Test cycle task update via import with correct data"""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            cycle_task = wf_factories.CycleTaskGroupObjectTaskFactory(
                description="task active description 1")
            self.import_data(
                collections.OrderedDict([
                    ("object_type", "Cycle Task Group Object Task"),
                    ("Code*", cycle_task.slug),
                    ("description", "task active description 1"),
                ]))

        self._cmp_tasks(self.expected_cycle_task_correct)

    def test_cycle_task_warnings(self):
        """Test cycle task update via import with data which is the reason of
    warnings about non-importable columns."""
        self._generate_cycle_tasks()

        expected_warnings = {
            'Cycle Task': {
                'block_warnings': {
                    errors.ONLY_IMPORTABLE_COLUMNS_WARNING.format(
                        line=2,
                        columns=", ".join(self.IMPORTABLE_COLUMN_NAMES))
                },
                'row_warnings': []
            }
        }

        with freeze_time(self.ftime_active):
            person = ggrc_factories.PersonFactory()
            cycle_task = wf_factories.CycleTaskGroupObjectTaskFactory(
                title="task active title134567890")
            ct_data = collections.OrderedDict([
                ("object_type", "Cycle Task Group Object Task"),
                ("Code*", cycle_task.slug),
                ("Summary*", "task active title1-Updated"),
                ("Task Description", "task active title1 description"),
                ("Start Date", self.ftime_historical),
                ("Due Date", self.ftime_active),
                ("Actual Finish Date", self.ftime_active),
                ("Actual Verified Date", self.ftime_active),
                ("State", "Assigned"),
                ("Task Assignees", person.email),
                ("Task Secondary Assignees", person.email),
            ])

            response = self.import_data(ct_data)
            self._check_csv_response(response, expected_warnings)
            self._cmp_tasks(self.expected_cycle_task_correct)

    def test_cycle_task_permission_error(self):
        """Test cycle task update via import with non-admin user which is the
    reason of error. Only admin can update cycle tasks via import."""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            _, creator = self.object_generator.generate_person(
                user_role="Creator")

            cycle_task = wf_factories.CycleTaskGroupObjectTaskFactory(
                description="task active description 1")
            ct_data = collections.OrderedDict([
                ("object_type", "Cycle Task Group Object Task"),
                ("Code*", cycle_task.slug),
            ])
            response = self.import_data(ct_data, person=creator, safe=False)
            self._check_csv_response(response, self.expected_permission_error)
            # Cycle tasks' data shouldn't be changed in test DB after import run from
            # non-admin user

            expected_cycle_task_permission_error = {}
            expected_cycle_task_permission_error.update(
                self.generated_cycle_tasks_active)
            expected_cycle_task_permission_error.update(
                self.generated_cycle_tasks_historical)
            self._cmp_tasks(expected_cycle_task_permission_error)

    def _cmp_tasks(self, expected_ctasks):
        """Compare tasks values from argument's list and test DB."""
        for ctask in db.session.query(CycleTaskGroupObjectTask).all():
            if ctask.title not in expected_ctasks:
                continue
            exp_task = expected_ctasks[ctask.title]
            for attr, val in exp_task.iteritems():
                self.assertEqual(
                    str(getattr(ctask, attr, None)), val,
                    "attr {} value for {} not expected".format(
                        attr, ctask.title))

    # pylint: disable=too-many-arguments
    def _activate_workflow(self, ftime, workflow, task_group, task_group_tasks,
                           random_object, cycle_tasks):
        """Helper which is responsible for active cycle-tasks creation"""
        with freeze_time(ftime):
            _, wf = self.wf_generator.generate_workflow(workflow)
            _, tg = self.wf_generator.generate_task_group(wf, task_group)
            for task in task_group_tasks:
                self.wf_generator.generate_task_group_task(tg, task)
            self.wf_generator.generate_task_group_object(tg, random_object)
            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)
            for exp_slug, exp_task in cycle_tasks.iteritems():
                obj = db.session.query(CycleTaskGroupObjectTask).filter_by(
                    slug=exp_slug).first()
                if exp_task["status"] == "Verified":
                    self.wf_generator.modify_object(obj,
                                                    {"status": "Finished"})
                self.wf_generator.modify_object(obj,
                                                {"status": exp_task["status"]})
        self._cmp_tasks(cycle_tasks)
        return cycle

    def _generate_cycle_tasks(self):
        """Helper which is responsible for test data creation"""
        self._activate_workflow(self.ftime_active, self.workflow_active,
                                self.task_group_active,
                                self.task_group_tasks_active,
                                self.random_objects[0],
                                self.generated_cycle_tasks_active)
        cycle = self._activate_workflow(self.ftime_historical,
                                        self.workflow_historical,
                                        self.task_group_historical,
                                        self.task_group_tasks_historical,
                                        self.random_objects[1],
                                        self.generated_cycle_tasks_historical)
        with freeze_time(self.ftime_historical):
            cycle = Cycle.query.get(cycle.id)
            self.wf_generator.modify_object(cycle, data={"is_current": False})

    def _create_test_cases_data(self):
        """Create test cases data: for object generation,
    expected data for checks"""
        def person_dict(person_id):
            """Return person data"""
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        wf_admin_role_id = {
            n: i
            for (
                i,
                n) in role.get_custom_roles_for(Workflow.__name__).iteritems()
        }['Admin']

        self.workflow_active = {
            "title":
            "workflow active title",
            "description":
            "workflow active description",
            "is_verification_needed":
            True,
            "access_control_list":
            [acl_helper.get_acl_json(wf_admin_role_id, self.person_1.id)],
            "notify_on_change":
            False,
        }

        self.task_group_active = {
            "title": "task group active title",
            "contact": person_dict(self.person_1.id),
        }

        self.task_group_tasks_active = [{
            "title":
            "task active title 1",
            "description":
            "task active description 1",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/01/2016",
            "end_date":
            "07/06/2016",
        }, {
            "title":
            "task active title 2",
            "description":
            "task active description 2",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/07/2016",
            "end_date":
            "07/12/2016",
        }, {
            "title":
            "task active title 3",
            "description":
            "task active description 3",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/13/2016",
            "end_date":
            "07/18/2016",
        }, {
            "title":
            "task active title 4",
            "description":
            "task active description 4",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/19/2016",
            "end_date":
            "07/24/2016",
        }, {
            "title":
            "task active title 5",
            "description":
            "task active description 5",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/25/2016",
            "end_date":
            "07/30/2016",
        }]

        # Active cycle tasks which should be generated from previous structure
        # at the beginning of each test
        self.generated_cycle_tasks_active = {
            "CYCLETASK-1": {
                "title": self.task_group_tasks_active[0]["title"],
                "description": self.task_group_tasks_active[0]["description"],
                "start_date": "2016-07-01",
                "end_date": "2016-07-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": self.task_group_tasks_active[1]["title"],
                "description": self.task_group_tasks_active[1]["description"],
                "start_date": "2016-07-07",
                "end_date": "2016-07-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-3": {
                "title": self.task_group_tasks_active[2]["title"],
                "description": self.task_group_tasks_active[2]["description"],
                "start_date": "2016-07-13",
                "end_date": "2016-07-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "In Progress"
            },
            "CYCLETASK-4": {
                "title": self.task_group_tasks_active[3]["title"],
                "description": self.task_group_tasks_active[3]["description"],
                "start_date": "2016-07-19",
                "end_date": "2016-07-22",
                "finished_date": "2016-07-01",
                "verified_date": "None",
                "status": "Finished"
            },
            "CYCLETASK-5": {
                "title": self.task_group_tasks_active[4]["title"],
                "description": self.task_group_tasks_active[4]["description"],
                "start_date": "2016-07-25",
                "end_date": "2016-07-29",
                "finished_date": "2016-07-01",
                "verified_date": "2016-07-01",
                "status": "Verified"
            }
        }

        self.workflow_historical = {
            "title":
            "workflow historical title",
            "description":
            "workflow historical description",
            "is_verification_needed":
            True,
            "access_control_list":
            [acl_helper.get_acl_json(wf_admin_role_id, self.person_1.id)],
            "notify_on_change":
            False,
        }

        self.task_group_historical = {
            "title": "task group historical title",
            "contact": person_dict(self.person_1.id),
        }

        self.task_group_tasks_historical = [
            {
                "title": "task historical title 1",
                "description": "task historical description 1",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/01/2014",
                "end_date": "05/06/2014",
            },
            {
                "title": "task historical title 2",
                "description": "task historical description 2",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/07/2014",
                "end_date": "05/12/2014",
            },
            {
                "title": "task historical title 3",
                "description": "task historical description 3",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/13/2014",
                "end_date": "05/18/2014",
            },
            {
                "title": "task historical title 4",
                "description": "task historical description 4",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/19/2014",
                "end_date": "05/24/2014",
            },
            {
                "title": "task historical title 5",
                "description": "task historical description 5",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/25/2014",
                "end_date": "05/30/2014",
            },
        ]

        # Historical cycle tasks which should be generated from previous structure
        # at the beginning of each test.
        self.generated_cycle_tasks_historical = {
            "CYCLETASK-6": {
                "title": self.task_group_tasks_historical[0]["title"],
                "description":
                self.task_group_tasks_historical[0]["description"],
                "start_date": "2014-05-01",
                "end_date": "2014-05-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-7": {
                "title": self.task_group_tasks_historical[1]["title"],
                "description":
                self.task_group_tasks_historical[1]["description"],
                "start_date": "2014-05-07",
                "end_date": "2014-05-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-8": {
                "title": self.task_group_tasks_historical[2]["title"],
                "description":
                self.task_group_tasks_historical[2]["description"],
                "start_date": "2014-05-13",
                "end_date": "2014-05-16",
                "finished_date": "None",
                "verified_date": "None",
                "status": "In Progress"
            },
            "CYCLETASK-9": {
                "title": self.task_group_tasks_historical[3]["title"],
                "description":
                self.task_group_tasks_historical[3]["description"],
                "start_date": "2014-05-19",
                "end_date": "2014-05-23",
                "finished_date": "2014-05-01",
                "verified_date": "None",
                "status": "Finished"
            },
            "CYCLETASK-10": {
                "title": self.task_group_tasks_historical[4]["title"],
                "description":
                self.task_group_tasks_historical[4]["description"],
                "start_date": "2014-05-23",
                "end_date": "2014-05-30",
                "finished_date": "2014-05-01",
                "verified_date": "2014-05-01",
                "status": "Verified"
            }
        }

        # Expected cycle tasks which should be created in correct cycle task update
        # case. It is needed for most tests.
        self.expected_cycle_task_correct = {
            "CYCLETASK-1": {
                "title": self.task_group_tasks_active[0]["title"] + " one",
                "description":
                self.task_group_tasks_active[0]["description"] + " one",
                "start_date": "2016-06-01",
                "end_date": "2016-06-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": self.task_group_tasks_active[1]["title"] + " two",
                "description":
                self.task_group_tasks_active[1]["description"] + " two",
                "start_date": "2016-06-07",
                "end_date": "2016-06-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-3": {
                "title": self.task_group_tasks_active[2]["title"] + " three",
                "description":
                self.task_group_tasks_active[2]["description"] + " three",
                "start_date": "2016-06-13",
                "end_date": "2016-06-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "In Progress"
            },
            "CYCLETASK-4": {
                "title": self.task_group_tasks_active[3]["title"] + " four",
                "description":
                self.task_group_tasks_active[3]["description"] + " four",
                "start_date": "2016-06-19",
                "end_date": "2016-06-24",
                "status": "Finished"
            },
            "CYCLETASK-5": {
                "title": self.task_group_tasks_active[4]["title"] + " five",
                "description":
                self.task_group_tasks_active[4]["description"] + " five",
                "start_date": "2016-06-25",
                "end_date": "2016-06-30",
                "finished_date": "2016-07-01",
                "verified_date": "2016-07-01",
                "status": "Verified"
            },
            "CYCLETASK-6": {
                "title":
                self.task_group_tasks_historical[0]["title"] + " one",
                "description":
                self.task_group_tasks_historical[0]["description"] + " one",
                "start_date":
                "2014-04-01",
                "end_date":
                "2014-04-06",
                "finished_date":
                "None",
                "verified_date":
                "None",
                "status":
                "Assigned"
            },
            "CYCLETASK-7": {
                "title":
                self.task_group_tasks_historical[1]["title"] + " two",
                "description":
                self.task_group_tasks_historical[1]["description"] + " two",
                "start_date":
                "2014-04-07",
                "end_date":
                "2014-04-12",
                "finished_date":
                "None",
                "verified_date":
                "None",
                "status":
                "Declined"
            },
            "CYCLETASK-8": {
                "title":
                self.task_group_tasks_historical[2]["title"] + " three",
                "description":
                self.task_group_tasks_historical[2]["description"] + " three",
                "start_date":
                "2014-04-13",
                "end_date":
                "2014-04-18",
                "finished_date":
                "None",
                "verified_date":
                "None",
                "status":
                "In Progress"
            },
            "CYCLETASK-9": {
                "title":
                self.task_group_tasks_historical[3]["title"] + " four",
                "description":
                self.task_group_tasks_historical[3]["description"] + " four",
                "start_date":
                "2014-04-19",
                "end_date":
                "2014-04-24",
                "status":
                "Finished"
            },
            "CYCLETASK-10": {
                "title":
                self.task_group_tasks_historical[4]["title"] + " five",
                "description":
                self.task_group_tasks_historical[4]["description"] + " five",
                "start_date":
                "2014-04-25",
                "end_date":
                "2014-04-30",
                "finished_date":
                "2014-05-01",
                "verified_date":
                "2014-05-01",
                "status":
                "Verified"
            }
        }

        # This is an error message which should be shown during
        # test_cycle_task_create_error test
        self.expected_create_error = {
            'Cycle Task': {
                'row_errors': {errors.CREATE_INSTANCE_ERROR.format(line=13)}
            }
        }

        # Below is expected date errors for test_cycle_task_date_error. They should
        # be shown during date validator's tests.
        self.expected_date_error = {
            'Cycle Task': {
                'row_errors': {
                    errors.INVALID_START_END_DATES.format(
                        line=3,
                        start_date="Start Date",
                        end_date="End Date",
                    ),
                    errors.INVALID_START_END_DATES.format(
                        line=8,
                        start_date="Start Date",
                        end_date="End Date",
                    ),
                },
            }
        }
        # Below is expected cycle-tasks data which should appear in test DB after
        # test_cycle_task_date_error run
        self.expected_cycle_task_date_error = dict()
        self.expected_cycle_task_date_error.update(
            self.generated_cycle_tasks_active)
        self.expected_cycle_task_date_error.update(
            self.generated_cycle_tasks_historical)
        self.expected_cycle_task_date_error["CYCLETASK-9"] = (
            self.expected_cycle_task_correct["CYCLETASK-9"])
        self.expected_cycle_task_date_error["CYCLETASK-10"] = (
            self.expected_cycle_task_correct["CYCLETASK-10"])

        # Expected error message which should be shown after
        # test_cycle_task_permission_error run
        self.expected_permission_error = {
            'Cycle Task': {
                'block_errors': {errors.PERMISSION_ERROR.format(line=2)}
            }
        }
Exemple #23
0
class TestOneTimeWorkflowNotification(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        TestCase.setUp(self)
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.random_people = self.object_generator.generate_random_people(
            user_role="Administrator")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    def test_one_time_wf_activate(self):
        def get_person(person_id):
            return db.session.query(Person).filter(
                Person.id == person_id).one()

        with freeze_time("2015-04-10"):
            _, wf = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)

            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

            person_1 = get_person(self.random_people[0].id)

        with freeze_time("2015-04-11"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn("cycle_started", notif_data[person_1.email])
            self.assertIn(cycle.id,
                          notif_data[person_1.email]["cycle_started"])
            self.assertIn("my_tasks",
                          notif_data[person_1.email]["cycle_data"][cycle.id])

        with freeze_time("2015-05-03"):  # two days befor due date
            _, notif_data = common.get_daily_notifications()
            self.assertIn(person_1.email, notif_data)
            self.assertNotIn("due_in", notif_data[person_1.email])
            self.assertNotIn("due_today", notif_data[person_1.email])

        with freeze_time("2015-05-04"):  # one day befor due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[person_1.email]["due_in"]), 1)

        with freeze_time("2015-05-05"):  # due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[person_1.email]["due_today"]), 1)

    @patch("ggrc.notifications.common.send_email")
    def test_one_time_wf_activate_single_person(self, mock_mail):

        with freeze_time("2015-04-10"):
            user = "******"
            _, wf = self.wf_generator.generate_workflow(
                self.one_time_workflow_single_person)

            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

        with freeze_time("2015-04-11"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn("cycle_started", notif_data[user])
            self.assertIn(cycle.id, notif_data[user]["cycle_started"])
            self.assertIn("my_tasks", notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("cycle_tasks",
                          notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("my_task_groups",
                          notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("cycle_url",
                          notif_data[user]["cycle_started"][cycle.id])

            cycle = Cycle.query.get(cycle.id)
            cycle_data = notif_data[user]["cycle_data"][cycle.id]
            for task in cycle.cycle_task_group_object_tasks:
                self.assertIn(task.id, cycle_data["my_tasks"])
                self.assertIn(task.id, cycle_data["cycle_tasks"])
                self.assertIn("title", cycle_data["my_tasks"][task.id])
                self.assertIn("title", cycle_data["cycle_tasks"][task.id])
                self.assertIn("cycle_task_url",
                              cycle_data["cycle_tasks"][task.id])

        with freeze_time("2015-05-03"):  # two days before due date
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user, notif_data)
            self.assertNotIn("due_in", notif_data[user])
            self.assertNotIn("due_today", notif_data[user])

        with freeze_time("2015-05-04"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[user]["due_in"]), 2)

        with freeze_time("2015-05-05"):  # due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[user]["due_today"]), 2)

            common.send_daily_digest_notifications()
            self.assertEqual(mock_mail.call_count, 1)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "description":
            "some test workflow",
            "notify_on_change":
            True,
            "owners": [person_dict(self.random_people[3].id)],
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "contact":
                    person_dict(self.random_people[2].id),
                    "task_group_tasks": [
                        {
                            "title": "task 1",
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        },
                        {
                            "title": "task 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[1].id),
                            "start_date": date(2015, 5, 4),
                            "end_date": date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "contact":
                    person_dict(self.random_people[2].id),
                    "task_group_tasks": [
                        {
                            "title": "task 1 in tg 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 8),  # friday
                            "end_date": date(2015, 5, 12),
                        },
                        {
                            "title": "task 2 in tg 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[2].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }

        user = Person.query.filter(Person.email == "*****@*****.**").one().id

        self.one_time_workflow_single_person = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "owners": [person_dict(user)],
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "contact":
                    person_dict(user),
                    "task_group_tasks": [
                        {
                            "title":
                            u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                            "description": "some task. ",
                            "contact": person_dict(user),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        },
                        {
                            "title": "task 2",
                            "description": "some task",
                            "contact": person_dict(user),
                            "start_date": date(2015, 5, 4),
                            "end_date": date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "contact":
                    person_dict(user),
                    "task_group_tasks": [
                        {
                            "title":
                            u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                            "description": "some task",
                            "contact": person_dict(user),
                            "start_date": date(2015, 5, 8),  # friday
                            "end_date": date(2015, 5, 12),
                        },
                        {
                            "title": "task 2 in tg 2",
                            "description": "some task",
                            "contact": person_dict(user),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }
Exemple #24
0
class TestOneTimeWorkflowNotification(TestCase):
    """ Tests are defined in the g-sheet test grid under:
    WF EMAILS for unit tests (middle level)
  """
    def setUp(self):
        super(TestOneTimeWorkflowNotification, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.random_people = [
            self.object_generator.generate_person(user_role="Administrator")[1]
            for _ in range(5)
        ]
        self.create_test_cases()

        self.create_users()

        db.session.query(Notification).delete()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    def tearDown(self):
        db.session.query(Notification).delete()

    def short_dict(self, obj, plural):
        return {
            "href": "/api/%s/%d" % (plural, obj.id),
            "id": obj.id,
            "type": obj.__class__.__name__,
        }

    def test_one_time_wf(self):
        # setup
        with freeze_time("2015-04-07 03:21:34"):
            wf_response, wf = self.wf_generator.generate_workflow(
                data={
                    "owners":
                    None,  # owner will be the current user
                    "notify_on_change":
                    True,  # force real time updates
                    "title":
                    "One-time WF",
                    "notify_custom_message":
                    textwrap.dedent("""\
              Hi all.
              Did you know that Irelnd city namd Newtownmountkennedy has 19
              letters? But it's not the longest one. The recordsman is the
              city in New Zealand that contains 97 letter."""),
                })

            _, tg = self.wf_generator.generate_task_group(
                wf,
                data={
                    "title": "TG #1 for the One-time WF",
                    "contact": self.short_dict(self.tgassignee1, "people"),
                })

            self.wf_generator.generate_task_group_task(
                tg, {
                    "title": "task #1 for one-time workflow",
                    "contact": self.short_dict(self.member1, "people"),
                    "start_date": "04/07/2015",
                    "end_date": "04/15/2015",
                })

            self.wf_generator.generate_task_group_object(
                tg, self.random_objects[0])
            self.wf_generator.generate_task_group_object(
                tg, self.random_objects[1])

        # test
        with freeze_time("2015-04-07 03:21:34"):
            cycle_response, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

            common.get_daily_notifications()

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "description":
            "some test workflow",
            "owners": [person_dict(self.random_people[3].id)],
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "task_group_tasks": [
                        {
                            "title": "task 1",
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        },
                        {
                            "title": "task 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[1].id),
                            "start_date": date(2015, 5, 4),
                            "end_date": date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "task_group_tasks": [
                        {
                            "title": "task 1 in tg 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[0].id),
                            "start_date": date(2015, 5, 8),  # friday
                            "end_date": date(2015, 5, 12),
                        },
                        {
                            "title": "task 2 in tg 2",
                            "description": "some task",
                            "contact": person_dict(self.random_people[2].id),
                            "start_date": date(2015, 5, 1),  # friday
                            "end_date": date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }

    def create_users(self):
        _, self.owner1 = self.object_generator.generate_person(
            # data={"name": "User1 Owner1", "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.tgassignee1 = self.object_generator.generate_person(
            # data={"name": "User2 TGassignee1",
            #       "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.member1 = self.object_generator.generate_person(
            # data={"name": "User3 Member1", "email": "*****@*****.**"},
            user_role="Administrator")
        _, self.member2 = self.object_generator.generate_person(
            # data={"name": "User4 Member2", "email": "*****@*****.**"},
            user_role="Administrator")
class TestBasicWorkflowActions(TestCase):
  """
  Tests for basic workflow actions
  """
  def setUp(self):
    super(TestBasicWorkflowActions, self).setUp()
    self.api = Api()
    self.generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.create_test_cases()

  def tearDown(self):
    pass

  def test_create_workflows(self):
    _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
    self.assertIsInstance(wflow, Workflow)

    task_groups = db.session.query(TaskGroup)\
        .filter(TaskGroup.workflow_id == wflow.id).all()

    self.assertEqual(len(task_groups),
                     len(self.one_time_workflow_1["task_groups"]))

  def test_workflows(self):
    for workflow in self.all_workflows:
      _, wflow = self.generator.generate_workflow(workflow)
      self.assertIsInstance(wflow, Workflow)

      task_groups = db.session.query(TaskGroup)\
          .filter(TaskGroup.workflow_id == wflow.id).all()

      self.assertEqual(len(task_groups),
                       len(workflow["task_groups"]))

  def test_activate_wf(self):
    for workflow in self.all_workflows:
      _, wflow = self.generator.generate_workflow(workflow)
      response, wflow = self.generator.activate_workflow(wflow)

      self.assert200(response)

  def test_one_time_workflow_edits(self):
    _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)

    wf_dict = {"title": "modified one time wf"}
    self.generator.modify_workflow(wflow, data=wf_dict)

    modified_wf = db.session.query(Workflow).filter(
        Workflow.id == wflow.id).one()
    self.assertEqual(wf_dict["title"], modified_wf.title)

  def test_one_time_wf_activate(self):
    _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
    self.generator.generate_cycle(wflow)
    self.generator.activate_workflow(wflow)

    tasks = [len(tg.get("task_group_tasks", []))
             for tg in self.one_time_workflow_1["task_groups"]]

    cycle_tasks = db.session.query(CycleTaskGroupObjectTask).join(
        Cycle).join(Workflow).filter(Workflow.id == wflow.id).all()
    active_wf = db.session.query(Workflow).filter(
        Workflow.id == wflow.id).one()

    self.assertEqual(sum(tasks), len(cycle_tasks))
    self.assertEqual(active_wf.status, "Active")

  def test_one_time_wf_state_transition_dates(self):
    _, wflow = self.generator.generate_workflow(self.one_time_workflow_1)
    self.generator.generate_cycle(wflow)
    self.generator.activate_workflow(wflow)

    cycle_tasks = db.session.query(CycleTaskGroupObjectTask).join(
        Cycle).join(Workflow).filter(Workflow.id == wflow.id).all()
    with freeze_time("2015-6-9 13:00:00"):
      today = dtm.datetime.now()
      transitions = [
          ("InProgress", None, None),
          ("Finished", today, None),
          ("Declined", None, None),
          ("Finished", today, None),
          ("Verified", today, today),
          ("Finished", today, None),
      ]
      # Iterate over possible transitions and check if dates got set correctly
      for (status, expected_finished, expected_verified) in transitions:
        cycle_task = cycle_tasks[0]
        _, task = self.generator.modify_object(cycle_task, {"status": status})
        self.assertEqual(task.finished_date, expected_finished)
        self.assertEqual(task.verified_date, expected_verified)

  def test_delete_calls(self):
    _, workflow = self.generator.generate_workflow()
    self.generator.generate_task_group(workflow)
    _, task_group = self.generator.generate_task_group(workflow)
    task_groups = db.session.query(TaskGroup).filter(
        TaskGroup.workflow_id == workflow.id).all()
    self.assertEqual(len(task_groups), 2)

    response = self.generator.api.delete(task_group)
    self.assert200(response)

    task_groups = db.session.query(TaskGroup).filter(
        TaskGroup.workflow_id == workflow.id).all()
    self.assertEqual(len(task_groups), 1)

  def create_test_cases(self):

    self.quarterly_wf_1 = {
        "title": "quarterly wf 1",
        "description": "",
        "unit": "month",
        "repeat_every": 3,
        "task_groups": [{
            "title": "tg_1",
            "task_group_tasks": [{
                "description": factories.random_str(100),
            }, {
                "description": factories.random_str(100),
            }, {
                "description": factories.random_str(100),
            },
            ],
        },
        ]
    }

    self.weekly_wf_1 = {
        "title": "weekly thingy",
        "description": "start this many a time",
        "unit": "week",
        "repeat_every": 1,
        "task_groups": [{
            "title": "tg_2",
            "task_group_tasks": [{
                "description": factories.random_str(100),
            }, {
                "title": "monday task",
            }, {
                "title": "weekend task",
            },
            ],
            "task_group_objects": self.random_objects
        },
        ]
    }

    self.one_time_workflow_1 = {
        "title": "one time wf test",
        "description": "some test workflow",
        "task_groups": [{
            "title": "tg_1",
            "task_group_tasks": [{}, {}, {}]
        }, {
            "title": "tg_2",
            "task_group_tasks": [{
                "description": factories.random_str(100)
            }, {}
            ],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "tg_3",
            "task_group_tasks": [{
                "title": "simple task 1",
                "description": factories.random_str(100)
            }, {
                "title": factories.random_str(),
                "description": factories.random_str(100)
            }, {
                "title": factories.random_str(),
                "description": factories.random_str(100)
            }
            ],
            "task_group_objects": self.random_objects
        }
        ]
    }
    self.one_time_workflow_2 = {
        "title": "test_wf_title",
        "description": "some test workflow",
        "task_groups": [
            {
                "title": "tg_1",
                "task_group_tasks": [{}, {}, {}]
            },
            {
                "title": "tg_2",
                "task_group_tasks": [{
                    "description": factories.random_str(100)
                }, {}],
                "task_group_objects": self.random_objects[:2]
            },
            {
                "title": "tg_3",
                "task_group_tasks": [{
                    "title": "simple task 1",
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }, {
                    "title": factories.random_str(),
                    "description": factories.random_str(100)
                }],
                "task_group_objects": []
            }
        ]
    }

    self.monthly_workflow_1 = {
        "title": "monthly test wf",
        "description": "start this many a time",
        "unit": "month",
        "repeat_every": 1,
        "task_groups": [
            {
                "title": "tg_2",
                "task_group_tasks": [{
                    "description": factories.random_str(100),
                }, {
                    "title": "monday task",
                }, {
                    "title": "weekend task",
                }],
                "task_group_objects": self.random_objects
            },
        ]
    }
    self.all_workflows = [
        self.one_time_workflow_1,
        self.one_time_workflow_2,
        self.weekly_wf_1,
        self.monthly_workflow_1,
        self.quarterly_wf_1,
    ]
Exemple #26
0
class TestEnableAndDisableNotifications(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        super(TestEnableAndDisableNotifications, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        models.Notification.query.delete()

        self.random_objects = self.object_generator.generate_random_objects(2)
        _, self.user = self.object_generator.generate_person(
            user_role="Administrator")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        models.Notification.__init__ = init_decorator(
            models.Notification.__init__)

    @patch("ggrc.notifications.common.send_email")
    def test_default_notifications_settings(self, mock_mail):

        with freeze_time("2015-02-01 13:39:20"):
            _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
            response, wf = self.wf_generator.activate_workflow(wf)

            self.assert200(response)

            user = models.Person.query.get(self.user.id)

        with freeze_time("2015-01-01 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(user.email, notif_data)

        with freeze_time("2015-01-29 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_disabled_notifications(self, mock_mail):

        with freeze_time("2015-02-01 13:39:20"):
            _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
            response, wf = self.wf_generator.activate_workflow(wf)

            self.assert200(response)

            self.object_generator.generate_notification_setting(
                self.user.id, "Email_Digest", False)

            user = models.Person.query.get(self.user.id)

        with freeze_time("2015-01-01 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(user.email, notif_data)

        with freeze_time("2015-01-29 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(user.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_enabled_notifications(self, mock_mail):

        with freeze_time("2015-02-01 13:39:20"):
            _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
            response, wf = self.wf_generator.activate_workflow(wf)
            self.assert200(response)

        with freeze_time("2015-01-29 13:39:20"):
            user = models.Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)

            self.object_generator.generate_notification_setting(
                self.user.id, "Email_Digest", True)

            user = models.Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_forced_notifications(self, mock_mail):

        with freeze_time("2015-02-01 13:39:20"):
            _, wf = self.wf_generator.generate_workflow(
                self.quarterly_wf_forced)
            response, wf = self.wf_generator.activate_workflow(wf)

            self.assert200(response)

            user = models.Person.query.get(self.user.id)

        with freeze_time("2015-01-29 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)

            self.object_generator.generate_notification_setting(
                self.user.id, "Email_Digest", True)

            user = models.Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_force_one_wf_notifications(self, mock_mail):

        with freeze_time("2015-02-01 13:39:20"):
            _, wf_forced = self.wf_generator.generate_workflow(
                self.quarterly_wf_forced)
            response, wf_forced = self.wf_generator.activate_workflow(
                wf_forced)
            _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
            response, wf = self.wf_generator.activate_workflow(wf)

            self.assert200(response)

            user = models.Person.query.get(self.user.id)

        with freeze_time("2015-01-29 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)
            self.assertIn("cycle_starts_in", notif_data[user.email])
            self.assertIn(wf_forced.id,
                          notif_data[user.email]["cycle_starts_in"])
            self.assertIn(wf.id, notif_data[user.email]["cycle_starts_in"])

            self.object_generator.generate_notification_setting(
                self.user.id, "Email_Digest", False)

            user = models.Person.query.get(self.user.id)
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user.email, notif_data)
            self.assertIn("cycle_starts_in", notif_data[user.email])
            self.assertIn(wf_forced.id,
                          notif_data[user.email]["cycle_starts_in"])
            self.assertNotIn(wf.id, notif_data[user.email]["cycle_starts_in"])

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.quarterly_wf_forced = {
            "title":
            "quarterly wf forced notifications",
            "notify_on_change":
            True,
            "description":
            "",
            # admin will be current user with id == 1
            "unit":
            "month",
            "repeat_every":
            3,
            "task_groups": [
                {
                    "title":
                    "tg_1",
                    "contact":
                    person_dict(self.user.id),
                    "task_group_tasks": [
                        {
                            "contact": person_dict(self.user.id),
                            "description": factories.random_str(100),
                        },
                    ],
                },
            ]
        }

        self.quarterly_wf = {
            "title":
            "quarterly wf 1",
            "description":
            "",
            # admin will be current user with id == 1
            "unit":
            "month",
            "repeat_every":
            3,
            "task_groups": [
                {
                    "title":
                    "tg_1",
                    "contact":
                    person_dict(self.user.id),
                    "task_group_tasks": [
                        {
                            "contact": person_dict(self.user.id),
                            "description": factories.random_str(100),
                        },
                    ],
                },
            ]
        }
class TestTaskDueNotifications(TestCase):
  """Test suite for task due soon/today notifications."""

  # pylint: disable=invalid-name

  def _fix_notification_init(self):
    """Fix Notification object init function.

    This is a fix needed for correct created_at field when using freezgun. By
    default the created_at field is left empty and filed by database, which
    uses system time and not the fake date set by freezugun plugin. This fix
    makes sure that object created in freeze_time block has all dates set with
    the correct date and time.
    """
    def init_decorator(init):
      """"Adjust the value of the object's created_at attribute to now."""
      @functools.wraps(init)
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    models.Notification.__init__ = init_decorator(models.Notification.__init__)

  def setUp(self):
    super(TestTaskDueNotifications, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    models.Notification.query.delete()

    self._fix_notification_init()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")

    role_id = models.all_models.AccessControlRole.query.filter(
        models.all_models.AccessControlRole.name == "Task Assignees",
        models.all_models.AccessControlRole.object_type == "TaskGroupTask",
    ).one().id

    self.one_time_workflow = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "is_verification_needed": False,
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "contact": {
                "href": "/api/people/{}".format(self.user.id),
                "id": self.user.id,
                "type": "Person",
            },
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
                "start_date": date(2017, 5, 15),
                "end_date": date(2017, 6, 11),
            }, {
                "title": "task 2",
                "description": "some task 2",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
                "start_date": date(2017, 5, 8),
                "end_date": date(2017, 6, 12),
            }, {
                "title": "task 3",
                "description": "some task 3",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
                "start_date": date(2017, 5, 31),
                "end_date": date(2017, 6, 13),
            }, {
                "title": "task 4",
                "description": "some task 4",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
                "start_date": date(2017, 6, 2),
                "end_date": date(2017, 6, 14),
            }, {
                "title": "task 5",
                "description": "some task 5",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
                "start_date": date(2017, 6, 8),
                "end_date": date(2017, 6, 15),
            }],
            "task_group_objects": self.random_objects
        }]
    }

  @ddt.unpack
  @ddt.data(
      ("2017-06-12 12:12:12", ["task 1"], ["task 2"], ["task 3"]),
      ("2017-06-13 13:13:13", ["task 1", "task 2"], ["task 3"], ["task 4"]),
  )
  @patch("ggrc.notifications.common.send_email")
  def test_creating_obsolete_notifications(
      self, fake_now, expected_overdue, expected_due_today, expected_due_in, _
  ):
    """Notifications already obsolete on creation date should not be created.
    """
    with freeze_time("2017-06-12 09:39:32"):
      tmp = self.one_time_workflow.copy()
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    user = models.Person.query.get(self.user.id)

    with freeze_time(fake_now):
      # mark all yeasterday notifications as sent
      models.all_models.Notification.query.filter(
          sa.func.DATE(models.all_models.Notification.send_on) < date.today()
      ).update({models.all_models.Notification.sent_at:
                datetime.now() - timedelta(1)},
               synchronize_session="fetch")

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})

      actual_overdue = [n['title'] for n in
                        user_notifs.get("task_overdue", {}).itervalues()]
      actual_overdue.sort()
      self.assertEqual(actual_overdue, expected_overdue)

      self.assertEqual(
          [n['title'] for n in user_notifs.get("due_today", {}).itervalues()],
          expected_due_today)

      self.assertEqual(
          [n['title'] for n in user_notifs.get("due_in", {}).itervalues()],
          expected_due_in)
class TestNotificationsForDeletedObjects(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestNotificationsForDeletedObjects, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_delete_activated_workflow(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, workflow = self.wf_generator.generate_workflow(self.quarterly_wf_1)
      response, workflow = self.wf_generator.activate_workflow(workflow)

      self.assert200(response)

      user = Person.query.get(self.user.id)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_starts_in", notif_data[user.email])

      workflow = Workflow.query.get(workflow.id)

      response = self.wf_generator.api.delete(workflow)
      self.assert200(response)

      _, notif_data = common.get_daily_notifications()
      user = Person.query.get(self.user.id)

      self.assertNotIn(user.email, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.quarterly_wf_1 = {
        "title": "quarterly wf 1",
        "notify_on_change": True,
        "description": "",
        "owners": [person_dict(self.user.id)],
        "unit": "month",
        "repeat_every": 3,
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
            },
            ],
        },
        ]
    }
class TestCycleStartFailed(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        TestCase.setUp(self)
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        Notification.query.delete()

        self.random_objects = self.object_generator.generate_random_objects(2)
        _, self.user = self.object_generator.generate_person(
            user_role="Administrator")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    @patch("ggrc.notifications.common.send_email")
    def test_start_failed(self, mock_mail):

        wf_owner = "*****@*****.**"

        with freeze_time("2015-02-01 13:39:20"):
            _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
            response, wf = self.wf_generator.activate_workflow(wf)
            print wf.next_cycle_start_date

            self.assert200(response)

        with freeze_time("2015-01-01 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(wf_owner, notif_data)

        with freeze_time("2015-01-29 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(wf_owner, notif_data)
            self.assertIn("cycle_starts_in", notif_data[wf_owner])

        with freeze_time("2015-03-05 13:39:20"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(wf_owner, notif_data)
            self.assertNotIn("cycle_started", notif_data[wf_owner])
            self.assertIn(wf_owner, notif_data)
            self.assertIn("cycle_start_failed", notif_data[wf_owner])

            common.send_daily_digest_notifications()

            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(wf_owner, notif_data)

    # TODO: investigate why next_cycle_start date remains the same after
    # start_recurring_cycles

    # @patch("ggrc.notifications.common.send_email")
    # def test_start_failed_send_notifications(self, mock_mail):

    #   wf_owner = "*****@*****.**"

    #   with freeze_time("2015-02-01 13:39:20"):
    #     _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
    #     response, wf = self.wf_generator.activate_workflow(wf)
    #     print wf.next_cycle_start_date

    #     self.assert200(response)

    #   with freeze_time("2015-01-01 13:39:20"):
    #     _, notif_data = common.get_daily_notifications()
    #     self.assertNotIn(wf_owner, notif_data)

    #   with freeze_time("2015-01-29 13:39:20"):
    #     _, notif_data = common.get_daily_notifications()
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertIn("cycle_starts_in", notif_data[wf_owner])

    #   with freeze_time("2015-02-05 13:39:20"):
    #     _, notif_data = common.get_daily_notifications()
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertNotIn("cycle_started", notif_data[wf_owner])
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertIn("cycle_start_failed", notif_data[wf_owner])

    #     start_recurring_cycles()
    #     _, notif_data = common.get_daily_notifications()
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertIn("cycle_started", notif_data[wf_owner])
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertNotIn("cycle_start_failed", notif_data[wf_owner])

    #     common.send_daily_digest_notifications()

    #     _, notif_data = common.get_daily_notifications()
    #     self.assertNotIn(wf_owner, notif_data)

    # @patch("ggrc.notifications.common.send_email")
    # def test_start_failed_send_notifications_monthly(self, mock_mail):

    #   wf_owner = "*****@*****.**"

    #   with freeze_time("2015-05-12 13:39:20"):
    #     _, wf = self.wf_generator.generate_workflow(self.monthly)
    #     response, wf = self.wf_generator.activate_workflow(wf)

    #   with freeze_time("2015-05-14 13:39:20"):
    #     _, wf = self.wf_generator.generate_workflow(self.monthly)
    #     response, wf = self.wf_generator.activate_workflow(wf)

    #     self.assert200(response)

    #     _, notif_data = common.get_daily_notifications()
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertNotIn("cycle_started", notif_data[wf_owner])
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertIn("cycle_start_failed", notif_data[wf_owner])

    #     start_recurring_cycles()
    #     _, notif_data = common.get_daily_notifications()
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertIn("cycle_started", notif_data[wf_owner])
    #     self.assertIn(wf_owner, notif_data)
    #     self.assertNotIn("cycle_start_failed", notif_data[wf_owner])

    #     common.send_daily_digest_notifications()

    #     _, notif_data = common.get_daily_notifications()
    #     self.assertNotIn(wf_owner, notif_data)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.quarterly_wf = {
            "title":
            "quarterly wf forced notifications",
            "notify_on_change":
            True,
            "description":
            "",
            "owners": [person_dict(self.user.id)],
            "frequency":
            "quarterly",
            "task_groups": [
                {
                    "title":
                    "tg_1",
                    "contact":
                    person_dict(self.user.id),
                    "task_group_tasks": [
                        {
                            "contact": person_dict(self.user.id),
                            "description": self.wf_generator.random_str(100),
                            "relative_start_day": 5,
                            "relative_start_month": 2,
                            "relative_end_day": 25,
                            "relative_end_month": 2,
                        },
                    ],
                },
            ]
        }
        self.monthly = {
            "title":
            "monthly",
            "notify_on_change":
            True,
            "description":
            "",
            "owners": [person_dict(self.user.id)],
            "frequency":
            "monthly",
            "task_groups": [
                {
                    "title":
                    "tg_1",
                    "contact":
                    person_dict(self.user.id),
                    "task_group_tasks": [
                        {
                            "contact": person_dict(self.user.id),
                            "description": self.wf_generator.random_str(100),
                            "relative_start_day": 14,
                            "relative_end_day": 25,
                        },
                    ],
                },
            ]
        }
Exemple #30
0
class TestRecurringWorkflowRevisions(TestCase):
    """Starting start recurring cycle should generate revisions."""
    def setUp(self):
        super(TestRecurringWorkflowRevisions, self).setUp()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        _, self.person_1 = self.object_generator.generate_person(
            user_role="Administrator")
        _, self.person_2 = self.object_generator.generate_person(
            user_role="Administrator")

        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.monthly_workflow = {
            "title":
            "test monthly wf notifications",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            # admin will be user with id == 1
            "unit":
            "month",
            "repeat_every":
            1,
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title": "task 1",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                }, {
                    "title": "task 2",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                }],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "another one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title": "task 1 in tg 2",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                }, {
                    "title": "task 2 in tg 2",
                    "description": "some task",
                    "contact": person_dict(self.person_2.id),
                }],
                "task_group_objects": []
            }]
        }

    @unittest.skip("Required to fix log event procedure for new calculator")
    @patch("ggrc.notifications.common.send_email")
    def test_revisions(self, mock_mail):  # pylint: disable=unused-argument
        with freeze_time("2015-04-01"):
            _, workflow = self.wf_generator.generate_workflow(
                self.monthly_workflow)
            self.wf_generator.activate_workflow(workflow)
        event_count = Event.query.count()
        revision_query = Revision.query.filter_by(
            resource_type='CycleTaskGroupObjectTask', )
        revision_count = revision_query.count()
        # cycle starts on monday - 6th, and not on 5th
        with freeze_time("2015-04-03"):
            start_recurring_cycles()

        self.assertEqual(event_count + 1, Event.query.count())
        self.assertNotEqual(revision_count, revision_query.count())
class TestTaskOverdueNotificationsUsingAPI(TestTaskOverdueNotifications):
  """Tests for overdue notifications when changing Tasks with an API."""

  # pylint: disable=invalid-name

  def setUp(self):
    super(TestTaskOverdueNotificationsUsingAPI, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    models.Notification.query.delete()

    self._fix_notification_init()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self._create_test_cases()

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_sending_overdue_notifications_for_tasks(self, is_vf_needed, _):
    """Overdue notifications should be sent for overdue tasks every day.

    Even if an overdue notification has already been sent, it should still be
    sent in every following daily digest f a task is still overdue.
    """
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    tasks = workflow.cycles[0].cycle_task_group_object_tasks
    task1_id = tasks[0].id
    task2_id = tasks[1].id

    user = models.Person.query.get(self.user.id)

    with freeze_time("2017-05-14 08:09:10"):
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

    with freeze_time("2017-05-15 08:09:10"):  # task 1 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

    with freeze_time("2017-05-16 08:09:10"):  # task 2 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id])

    with freeze_time("2017-05-17 08:09:10"):  # after both tasks' due dates
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id, task2_id])

      common.send_daily_digest_notifications()

    # even after sending the overdue notifications, they are sent again the
    # day after, too
    with freeze_time("2017-05-18 08:09:10"):
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id, task2_id])

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_adjust_overdue_notifications_on_task_due_date_change(self,
                                                                is_vf_needed,
                                                                _):
    """Sending overdue notifications should adjust to task due date changes."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

      tasks = workflow.cycles[0].cycle_task_group_object_tasks
      task1, task2 = tasks
      self.wf_generator.modify_object(task2, {"end_date": date(2099, 12, 31)})

      user = models.Person.query.get(self.user.id)

    with freeze_time("2017-05-16 08:09:10"):  # a day after task1 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

      # change task1 due date, there should be no overdue notification anymore
      self.wf_generator.modify_object(task1, {"end_date": date(2017, 5, 16)})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

      # change task1 due date to the past there should a notification again
      self.wf_generator.modify_object(task1, {"end_date": date(2017, 5, 14)})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_adjust_overdue_notifications_on_task_status_change(self,
                                                              is_vf_needed,
                                                              _):
    """Sending overdue notifications should take task status into account."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

      tasks = workflow.cycles[0].cycle_task_group_object_tasks
      task1, task2 = tasks
      self.wf_generator.modify_object(task2, {"end_date": date(2099, 12, 31)})

      user = models.Person.query.get(self.user.id)
      user_email = user.email
    if is_vf_needed:
      non_final_states = [CycleTaskGroupObjectTask.ASSIGNED,
                          CycleTaskGroupObjectTask.IN_PROGRESS,
                          CycleTaskGroupObjectTask.FINISHED,
                          CycleTaskGroupObjectTask.DECLINED]
      final_state = CycleTaskGroupObjectTask.VERIFIED
    else:
      non_final_states = [CycleTaskGroupObjectTask.ASSIGNED,
                          CycleTaskGroupObjectTask.IN_PROGRESS]
      final_state = CycleTaskGroupObjectTask.FINISHED

    with freeze_time("2017-05-16 08:09:10"):  # a day after task1 due date
      for state in non_final_states:
        # clear all notifications before before changing the task status
        models.Notification.query.delete()
        _, notif_data = common.get_daily_notifications()
        self.assertEqual(notif_data, {})

        self.wf_generator.modify_object(task1, {"status": state})

        _, notif_data = common.get_daily_notifications()
        user_notifs = notif_data.get(user_email, {})
        self.assertIn("task_overdue", user_notifs)
        self.assertEqual(len(user_notifs["task_overdue"]), 1)

      # WITHOUT clearing the overdue notifications, move the task to "verified"
      # state, and the overdue notification should disappear.

      self.wf_generator.modify_object(task1, {"status": final_state})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertNotIn("task_overdue", user_notifs)

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_stop_sending_overdue_notification_if_task_gets_deleted(self,
                                                                  is_vf_needed,
                                                                  _):
    """Overdue notifications should not be sent for deleted tasks."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    tasks = workflow.cycles[0].cycle_task_group_object_tasks
    task1, task2 = tasks

    user = models.Person.query.get(self.user.id)
    user_email = user.email

    with freeze_time("2017-10-16 08:09:10"):  # long after both task due dates
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 2)

      db.session.delete(task2)
      db.session.commit()

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

      db.session.delete(task1)
      db.session.commit()

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

  def _create_test_cases(self):
    """Create configuration to use for generating a new workflow."""
    def person_dict(person_id):
      return {
          "href": "/api/people/" + str(person_id),
          "id": person_id,
          "type": "Person"
      }
    role_id = models.all_models.AccessControlRole.query.filter(
        models.all_models.AccessControlRole.name == "Task Assignees",
        models.all_models.AccessControlRole.object_type == "TaskGroupTask",
    ).one().id
    self.one_time_workflow = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "owners": [person_dict(self.user.id)],
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "start_date": date(2017, 5, 5),  # Friday
                "end_date": date(2017, 5, 15),
                "access_control_list": [{
                    "person": {"id": self.user.id, },
                    "ac_role_id": role_id,
                }],
            }, {
                "title": "task 2",
                "description": "some task 2",
                "start_date": date(2017, 5, 5),  # Friday
                "end_date": date(2017, 5, 16),
                "access_control_list": [{
                    "person": {"id": self.user.id, },
                    "ac_role_id": role_id,
                }],
            }],
            "task_group_objects": self.random_objects
        }]
    }
Exemple #32
0
class TestOneTimeWorkflowNotification(TestCase):

  """ Tests are defined in the g-sheet test grid under:
    WF EMAILS for unit tests (middle level)
  """

  def setUp(self):
    super(TestOneTimeWorkflowNotification, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.random_people = [
        self.object_generator.generate_person(user_role="Administrator")[1]
        for _ in range(5)]
    self.create_test_cases()

    self.create_users()

    db.session.query(Notification).delete()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  def tearDown(self):
    db.session.query(Notification).delete()

  def short_dict(self, obj, plural):
    return {
        "href": "/api/%s/%d" % (plural, obj.id),
        "id": obj.id,
        "type": obj.__class__.__name__,
    }

  def setup_cycle_tasks(self):
    """Prepare environment with couple of active cycle tasks."""
    with freeze_time("2018-11-01"):
      _, workflow = self.wf_generator.generate_workflow(
          self.one_time_workflow_1
      )
      self.wf_generator.generate_cycle(workflow)
      self.wf_generator.activate_workflow(workflow)
    return all_models.CycleTaskGroupObjectTask.query

  def assert_nofication_sent_with(self, text):
    """Assert if text exists in sent notification."""
    with mock.patch("ggrc.notifications.common.send_email") as send_email:
      self.client.get("/_notifications/send_daily_digest")
      _, _, content = send_email.call_args[0]
    self.assertIn(text, content)

  def assert_nofication_sent_without(self, text):
    """Assert if text doesn't exist in sent notification."""
    with mock.patch("ggrc.notifications.common.send_email") as send_email:
      self.client.get("/_notifications/send_daily_digest")
      _, _, content = send_email.call_args[0]
    self.assertNotIn(text, content)

  def test_one_time_wf(self):
    # setup
    with freeze_time("2015-04-07 03:21:34"):
      wf_response, wf = self.wf_generator.generate_workflow(data={
          # admin will be the current user
          "notify_on_change": True,  # force real time updates
          "title": "One-time WF",
          "notify_custom_message": textwrap.dedent("""\
              Hi all.
              Did you know that Irelnd city namd Newtownmountkennedy has 19
              letters? But it's not the longest one. The recordsman is the
              city in New Zealand that contains 97 letter."""),
      })

      _, tg = self.wf_generator.generate_task_group(wf, data={
          "title": "TG #1 for the One-time WF",
          "contact": self.short_dict(self.tgassignee1, "people"),
      })

      self.wf_generator.generate_task_group_task(tg, {
          "title": "task #1 for one-time workflow",
          "contact": self.short_dict(self.member1, "people"),
          "start_date": "04/07/2015",
          "end_date": "04/15/2015",
      })

      self.wf_generator.generate_task_group_object(tg, self.random_objects[0])
      self.wf_generator.generate_task_group_object(tg, self.random_objects[1])

    # test
    with freeze_time("2015-04-07 03:21:34"):
      cycle_response, cycle = self.wf_generator.generate_cycle(wf)
      self.wf_generator.activate_workflow(wf)

      common.get_daily_notifications()

  def test_deprecated_ct_acl_update(self):
    """Test if acl update for deprecated CT will not create notification."""
    cycle_task = self.setup_cycle_tasks().first()
    cycle_task_title = cycle_task.title
    response = self.api.put(cycle_task, {"status": "Deprecated"})
    self.assert200(response)
    self.assert_nofication_sent_without(cycle_task_title)

    task_assignee = all_models.AccessControlRole.query.filter_by(
        name="Task Assignees",
        object_type="CycleTaskGroupObjectTask",
    ).first()
    person = self.object_generator.generate_person(user_role="Creator")[1]
    response = self.api.put(
        cycle_task,
        {
            "access_control_list": [{
                "ac_role_id": task_assignee.id,
                "person": {
                    "id": person.id,
                    "type": "Person",
                }
            }]
        }
    )
    self.assert200(response)
    self.assert_nofication_sent_without(cycle_task_title)

  def test_restore_deprecated_ct(self):
    """Test notifications for CT which was restored from Deprecated."""
    cycle_task = self.setup_cycle_tasks().first()
    cycle_task_title = cycle_task.title

    self.assert_nofication_sent_with(cycle_task_title)

    response = self.api.put(cycle_task, {"status": "Deprecated"})
    self.assert200(response)
    self.assert_nofication_sent_without(cycle_task_title)

    response = self.api.put(cycle_task, {"status": "Assigned"})
    self.assert200(response)
    self.assert_nofication_sent_with(cycle_task_title)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.one_time_workflow_1 = {
        "title": "one time test workflow",
        "description": "some test workflow",
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "task_group_tasks": [{
                "title": "task_{}".format(str(uuid.uuid4())),
                "description": "some task",
                "contact": person_dict(self.random_people[0].id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task_{}".format(str(uuid.uuid4())),
                "description": "some task",
                "contact": person_dict(self.random_people[1].id),
                "start_date": date(2015, 5, 4),
                "end_date": date(2015, 5, 7),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "task_group_tasks": [{
                "title": "task_{}".format(str(uuid.uuid4())),
                "description": "some task",
                "contact": person_dict(self.random_people[0].id),
                "start_date": date(2015, 5, 8),  # friday
                "end_date": date(2015, 5, 12),
            }, {
                "title": "task_{}".format(str(uuid.uuid4())),
                "description": "some task",
                "contact": person_dict(self.random_people[2].id),
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": []
        }]
    }

  def create_users(self):
    _, self.admin1 = self.object_generator.generate_person(
        # data={"name": "User1 Admin1", "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.tgassignee1 = self.object_generator.generate_person(
        # data={"name": "User2 TGassignee1",
        #       "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.member1 = self.object_generator.generate_person(
        # data={"name": "User3 Member1", "email": "*****@*****.**"},
        user_role="Administrator")
    _, self.member2 = self.object_generator.generate_person(
        # data={"name": "User4 Member2", "email": "*****@*****.**"},
        user_role="Administrator")
Exemple #33
0
class TestEnableAndDisableNotifications(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestEnableAndDisableNotifications, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    models.Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    models.Notification.__init__ = init_decorator(models.Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_default_notifications_settings(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
      response, wf = self.wf_generator.activate_workflow(wf)

      self.assert200(response)

      user = models.Person.query.get(self.user.id)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

  @patch("ggrc.notifications.common.send_email")
  def test_disabled_notifications(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
      response, wf = self.wf_generator.activate_workflow(wf)

      self.assert200(response)

      self.object_generator.generate_notification_setting(
          self.user.id, "Email_Digest", False)

      user = models.Person.query.get(self.user.id)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

  @patch("ggrc.notifications.common.send_email")
  def test_enabled_notifications(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
      response, wf = self.wf_generator.activate_workflow(wf)
      self.assert200(response)

    with freeze_time("2015-01-29 13:39:20"):
      user = models.Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

      self.object_generator.generate_notification_setting(
          self.user.id, "Email_Digest", True)

      user = models.Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

  @patch("ggrc.notifications.common.send_email")
  def test_forced_notifications(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf_forced)
      response, wf = self.wf_generator.activate_workflow(wf)

      self.assert200(response)

      user = models.Person.query.get(self.user.id)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

      self.object_generator.generate_notification_setting(
          self.user.id, "Email_Digest", True)

      user = models.Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)

  @patch("ggrc.notifications.common.send_email")
  def test_force_one_wf_notifications(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, wf_forced = self.wf_generator.generate_workflow(
          self.quarterly_wf_forced)
      response, wf_forced = self.wf_generator.activate_workflow(wf_forced)
      _, wf = self.wf_generator.generate_workflow(self.quarterly_wf)
      response, wf = self.wf_generator.activate_workflow(wf)

      self.assert200(response)

      user = models.Person.query.get(self.user.id)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_starts_in", notif_data[user.email])
      self.assertIn(wf_forced.id, notif_data[user.email]["cycle_starts_in"])
      self.assertIn(wf.id, notif_data[user.email]["cycle_starts_in"])

      self.object_generator.generate_notification_setting(
          self.user.id, "Email_Digest", False)

      user = models.Person.query.get(self.user.id)
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_starts_in", notif_data[user.email])
      self.assertIn(wf_forced.id, notif_data[user.email]["cycle_starts_in"])
      self.assertNotIn(wf.id, notif_data[user.email]["cycle_starts_in"])

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.quarterly_wf_forced = {
        "title": "quarterly wf forced notifications",
        "notify_on_change": True,
        "description": "",
        # admin will be current user with id == 1
        "unit": "month",
        "repeat_every": 3,
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
            },
            ],
        },
        ]
    }

    self.quarterly_wf = {
        "title": "quarterly wf 1",
        "description": "",
        # admin will be current user with id == 1
        "unit": "month",
        "repeat_every": 3,
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
            },
            ],
        },
        ]
    }
Exemple #34
0
class TestMonthlyWorkflowNotification(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        TestCase.setUp(self)
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        _, self.person_1 = self.object_generator.generate_person(
            user_role="Administrator")
        _, self.person_2 = self.object_generator.generate_person(
            user_role="Administrator")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        Notification.__init__ = init_decorator(Notification.__init__)

    @patch("ggrc.notifications.common.send_email")
    def test_auto_generate_cycle(self, mock_mail):

        with freeze_time("2015-04-01"):
            _, wf = self.wf_generator.generate_workflow(
                self.monthly_workflow_1)
            self.wf_generator.activate_workflow(wf)

            person_1 = Person.query.get(self.person_1.id)

        with freeze_time("2015-04-02"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn(person_1.email, notif_data)
            self.assertIn("cycle_starts_in", notif_data[person_1.email])

        with freeze_time("2015-04-02"):
            self.api.tc.get("nightly_cron_endpoint")
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(person_1.email, notif_data)

        with freeze_time("2015-04-02"):
            start_recurring_cycles()
            _, notif_data = common.get_daily_notifications()
            self.assertNotIn(person_1.email, notif_data)

        # cycle starts on monday - 6th, and not on 5th
        with freeze_time("2015-04-03"):
            start_recurring_cycles()

        with freeze_time("2015-04-15"):  # one day befor due date
            _, notif_data = common.get_daily_notifications()
            person_1 = Person.query.get(self.person_1.id)
            self.assertIn(person_1.email, notif_data)

        with freeze_time("2015-04-25"):  # due date
            _, notif_data = common.get_daily_notifications()
            person_1 = Person.query.get(self.person_1.id)
            self.assertIn(person_1.email, notif_data)

    @patch("ggrc.notifications.common.send_email")
    def test_manual_generate_cycle(self, mock_mail):

        with freeze_time("2015-04-01"):
            _, wf = self.wf_generator.generate_workflow(
                self.monthly_workflow_1)
            self.wf_generator.activate_workflow(wf)

            person_1 = Person.query.get(self.person_1.id)

        with freeze_time("2015-04-03"):
            _, notif_data = common.get_daily_notifications()

        with freeze_time("2015-04-03"):
            _, cycle = self.wf_generator.generate_cycle(wf)
            _, notif_data = common.get_daily_notifications()
            person_1 = Person.query.get(self.person_1.id)
            self.assertIn("cycle_started", notif_data[person_1.email])

        with freeze_time("2015-05-03"):  # two days befor due date
            _, notif_data = common.get_daily_notifications()
            person_1 = Person.query.get(self.person_1.id)
            self.assertIn(person_1.email, notif_data)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.monthly_workflow_1 = {
            "title":
            "test monthly wf notifications",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            "owners": [person_dict(self.person_2.id)],
            "frequency":
            "monthly",
            "task_groups": [{
                "title":
                "one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title": "task 1",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                    "relative_start_day": 5,
                    "relative_end_day": 25,
                }, {
                    "title": "task 2",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                    "relative_start_day": 10,
                    "relative_end_day": 21,
                }],
                "task_group_objects":
                self.random_objects[:2]
            }, {
                "title":
                "another one time task group",
                "contact":
                person_dict(self.person_1.id),
                "task_group_tasks": [{
                    "title": "task 1 in tg 2",
                    "description": "some task",
                    "contact": person_dict(self.person_1.id),
                    "relative_start_day": 15,
                    "relative_end_day": 15,
                }, {
                    "title": "task 2 in tg 2",
                    "description": "some task",
                    "contact": person_dict(self.person_2.id),
                    "relative_start_day": 15,
                    "relative_end_day": 28,
                }],
                "task_group_objects": []
            }]
        }
class TestCycleTaskImportUpdate(TestCase):
    """ This class contains simple cycle task update tests using import
  functionality
  """

    CSV_DIR = join(abspath(dirname(__file__)), "test_csvs/")

    def setUp(self):
        super(TestCycleTaskImportUpdate, self).setUp()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()
        self.random_objects = self.object_generator.generate_random_objects(2)
        _, self.person_1 = self.object_generator.generate_person(
            user_role="Administrator")
        self.ftime_active = "2016-07-01"
        self.ftime_historical = "2014-05-01"
        self._create_test_cases_data()
        # It is needed because cycle-tasks are generated automatically with
        # 'slug' based on auto_increment 'id' field.
        # At start of each test we suppose that created cycle-task's 'slug'
        # lie in range from 1 to 10.
        db.session.execute('ALTER TABLE cycle_task_group_object_tasks '
                           'AUTO_INCREMENT = 1')

    def test_cycle_task_correct(self):
        """Test cycle task update via import with correct data"""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            response = self.import_file("cycle_task_correct.csv")
            self._check_csv_response(response, {})
            self._cmp_tasks(self.expected_cycle_task_correct)

    def test_cycle_task_warnings(self):
        """Test cycle task update via import with data which is the reason of
    warnings about non-importable columns."""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            response = self.import_file("cycle_task_warnings.csv")
            self._check_csv_response(response, self.expected_warnings)
            self._cmp_tasks(self.expected_cycle_task_correct)

    def test_cycle_task_create_error(self):
        """Test cycle task update via import with data which is the reason of
    errors about new cycle task creation."""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            response = self.import_file("cycle_task_create_error.csv")
            self._check_csv_response(response, self.expected_create_error)
            self._cmp_tasks(self.expected_cycle_task_correct)

    def test_cycle_task_date_error(self):
        """Test cycle task update via import with data which is the reason of
    errors about incorrect dates in csv file."""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            response = self.import_file("cycle_task_date_error.csv")
            self._check_csv_response(response, self.expected_date_error)
            self._cmp_tasks(self.expected_cycle_task_date_error)

    def test_cycle_task_permission_error(self):
        """Test cycle task update via import with non-admin user which is the
    reason of error. Only admin can update cycle tasks via import."""
        self._generate_cycle_tasks()
        with freeze_time(self.ftime_active):
            _, creator = self.object_generator.generate_person(
                user_role="Creator")
            response = self.import_file("cycle_task_correct.csv",
                                        person=creator)
            self._check_csv_response(response, self.expected_permission_error)
            # Cycle tasks' data shouldn't be changed in test DB after import run from
            # non-admin user
            expected_cycle_task_permission_error = {}
            expected_cycle_task_permission_error.update(
                self.generated_cycle_tasks_active)
            expected_cycle_task_permission_error.update(
                self.generated_cycle_tasks_historical)
            self._cmp_tasks(expected_cycle_task_permission_error)

    def _cmp_tasks(self, expected_ctasks):
        """Compare tasks values from argument's list and test DB."""
        for ctask in db.session.query(CycleTaskGroupObjectTask).all():
            if ctask.slug not in expected_ctasks:
                continue
            exp_task = expected_ctasks[ctask.slug]
            for attr, val in exp_task.iteritems():
                self.assertEqual(str(getattr(ctask, attr, None)), val)

    # pylint: disable=too-many-arguments
    def _activate_workflow(self, ftime, workflow, task_group, task_group_tasks,
                           random_object, cycle_tasks):
        """Helper which is responsible for active cycle-tasks creation"""
        with freeze_time(ftime):
            _, wf = self.wf_generator.generate_workflow(workflow)
            _, tg = self.wf_generator.generate_task_group(wf, task_group)
            for task in task_group_tasks:
                self.wf_generator.generate_task_group_task(tg, task)
            self.wf_generator.generate_task_group_object(tg, random_object)

            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

            for exp_slug, exp_task in cycle_tasks.iteritems():
                obj = db.session.query(CycleTaskGroupObjectTask).filter_by(
                    slug=exp_slug).first()
                if exp_task["status"] == "Verified":
                    self.wf_generator.modify_object(obj,
                                                    {"status": "Finished"})
                self.wf_generator.modify_object(obj,
                                                {"status": exp_task["status"]})
        self._cmp_tasks(cycle_tasks)
        return cycle

    def _generate_cycle_tasks(self):
        """Helper which is responsible for test data creation"""
        self._activate_workflow(self.ftime_active, self.workflow_active,
                                self.task_group_active,
                                self.task_group_tasks_active,
                                self.random_objects[0],
                                self.generated_cycle_tasks_active)
        cycle = self._activate_workflow(self.ftime_historical,
                                        self.workflow_historical,
                                        self.task_group_historical,
                                        self.task_group_tasks_historical,
                                        self.random_objects[1],
                                        self.generated_cycle_tasks_historical)
        with freeze_time(self.ftime_historical):
            cycle = Cycle.query.get(cycle.id)
            self.wf_generator.modify_object(cycle, data={"is_current": False})

    def _create_test_cases_data(self):
        """Create test cases data: for object generation,
    expected data for checks"""
        def person_dict(person_id):
            """Return person data"""
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        self.workflow_active = {
            "title": "workflow active title",
            "description": "workflow active description",
            "frequency": "one_time",
            "owners": [person_dict(self.person_1.id)],
            "notify_on_change": False,
        }

        self.task_group_active = {
            "title": "task group active title",
            "contact": person_dict(self.person_1.id),
        }

        self.task_group_tasks_active = [{
            "title":
            "task active title 1",
            "description":
            "task active description 1",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/01/2016",
            "end_date":
            "07/06/2016",
        }, {
            "title":
            "task active title 2",
            "description":
            "task active description 2",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/07/2016",
            "end_date":
            "07/12/2016",
        }, {
            "title":
            "task active title 3",
            "description":
            "task active description 3",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/13/2016",
            "end_date":
            "07/18/2016",
        }, {
            "title":
            "task active title 4",
            "description":
            "task active description 4",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/19/2016",
            "end_date":
            "07/24/2016",
        }, {
            "title":
            "task active title 5",
            "description":
            "task active description 5",
            "contact":
            person_dict(self.person_1.id),
            "start_date":
            "07/25/2016",
            "end_date":
            "07/30/2016",
        }]

        # Active cycle tasks which should be generated from previous structure
        # at the beginning of each test
        self.generated_cycle_tasks_active = {
            "CYCLETASK-1": {
                "title": self.task_group_tasks_active[0]["title"],
                "description": self.task_group_tasks_active[0]["description"],
                "start_date": "2016-07-01",
                "end_date": "2016-07-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": self.task_group_tasks_active[1]["title"],
                "description": self.task_group_tasks_active[1]["description"],
                "start_date": "2016-07-07",
                "end_date": "2016-07-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-3": {
                "title": self.task_group_tasks_active[2]["title"],
                "description": self.task_group_tasks_active[2]["description"],
                "start_date": "2016-07-13",
                "end_date": "2016-07-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "InProgress"
            },
            "CYCLETASK-4": {
                "title": self.task_group_tasks_active[3]["title"],
                "description": self.task_group_tasks_active[3]["description"],
                "start_date": "2016-07-19",
                "end_date": "2016-07-24",
                "finished_date": "2016-07-01 00:00:00",
                "verified_date": "None",
                "status": "Finished"
            },
            "CYCLETASK-5": {
                "title": self.task_group_tasks_active[4]["title"],
                "description": self.task_group_tasks_active[4]["description"],
                "start_date": "2016-07-25",
                "end_date": "2016-07-30",
                "finished_date": "2016-07-01 00:00:00",
                "verified_date": "2016-07-01 00:00:00",
                "status": "Verified"
            }
        }

        self.workflow_historical = {
            "title": "workflow historical title",
            "description": "workflow historical description",
            "frequency": "one_time",
            "owners": [person_dict(self.person_1.id)],
            "notify_on_change": False,
        }

        self.task_group_historical = {
            "title": "task group historical title",
            "contact": person_dict(self.person_1.id),
        }

        self.task_group_tasks_historical = [
            {
                "title": "task historical title 1",
                "description": "task historical description 1",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/01/2014",
                "end_date": "05/06/2014",
            },
            {
                "title": "task historical title 2",
                "description": "task historical description 2",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/07/2014",
                "end_date": "05/12/2014",
            },
            {
                "title": "task historical title 3",
                "description": "task historical description 3",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/13/2014",
                "end_date": "05/18/2014",
            },
            {
                "title": "task historical title 4",
                "description": "task historical description 4",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/19/2014",
                "end_date": "05/24/2014",
            },
            {
                "title": "task historical title 5",
                "description": "task historical description 5",
                "contact": person_dict(self.person_1.id),
                "start_date": "05/25/2014",
                "end_date": "05/30/2014",
            },
        ]

        # Historical cycle tasks which should be generated from previous structure
        # at the beginning of each test.
        self.generated_cycle_tasks_historical = {
            "CYCLETASK-6": {
                "title": self.task_group_tasks_historical[0]["title"],
                "description":
                self.task_group_tasks_historical[0]["description"],
                "start_date": "2014-05-01",
                "end_date": "2014-05-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-7": {
                "title": self.task_group_tasks_historical[1]["title"],
                "description":
                self.task_group_tasks_historical[1]["description"],
                "start_date": "2014-05-07",
                "end_date": "2014-05-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-8": {
                "title": self.task_group_tasks_historical[2]["title"],
                "description":
                self.task_group_tasks_historical[2]["description"],
                "start_date": "2014-05-13",
                "end_date": "2014-05-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "InProgress"
            },
            "CYCLETASK-9": {
                "title": self.task_group_tasks_historical[3]["title"],
                "description":
                self.task_group_tasks_historical[3]["description"],
                "start_date": "2014-05-19",
                "end_date": "2014-05-24",
                "finished_date": "2014-05-01 00:00:00",
                "verified_date": "None",
                "status": "Finished"
            },
            "CYCLETASK-10": {
                "title": self.task_group_tasks_historical[4]["title"],
                "description":
                self.task_group_tasks_historical[4]["description"],
                "start_date": "2014-05-25",
                "end_date": "2014-05-30",
                "finished_date": "2014-05-01 00:00:00",
                "verified_date": "2014-05-01 00:00:00",
                "status": "Verified"
            }
        }

        # Expected cycle tasks which should be created in correct cycle task update
        # case. It is needed for most tests.
        self.expected_cycle_task_correct = {
            "CYCLETASK-1": {
                "title": self.task_group_tasks_active[0]["title"] + " one",
                "description":
                self.task_group_tasks_active[0]["description"] + " one",
                "start_date": "2016-06-01",
                "end_date": "2016-06-06",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Assigned"
            },
            "CYCLETASK-2": {
                "title": self.task_group_tasks_active[1]["title"] + " two",
                "description":
                self.task_group_tasks_active[1]["description"] + " two",
                "start_date": "2016-06-07",
                "end_date": "2016-06-12",
                "finished_date": "None",
                "verified_date": "None",
                "status": "Declined"
            },
            "CYCLETASK-3": {
                "title": self.task_group_tasks_active[2]["title"] + " three",
                "description":
                self.task_group_tasks_active[2]["description"] + " three",
                "start_date": "2016-06-13",
                "end_date": "2016-06-18",
                "finished_date": "None",
                "verified_date": "None",
                "status": "InProgress"
            },
            "CYCLETASK-4": {
                "title": self.task_group_tasks_active[3]["title"] + " four",
                "description":
                self.task_group_tasks_active[3]["description"] + " four",
                "start_date": "2016-06-19",
                "end_date": "2016-06-24",
                "finished_date": "2016-07-19 00:00:00",
                "verified_date": "None",
                "status": "Finished"
            },
            "CYCLETASK-5": {
                "title": self.task_group_tasks_active[4]["title"] + " five",
                "description":
                self.task_group_tasks_active[4]["description"] + " five",
                "start_date": "2016-06-25",
                "end_date": "2016-06-30",
                "finished_date": "2016-07-25 00:00:00",
                "verified_date": "2016-08-30 00:00:00",
                "status": "Verified"
            },
            "CYCLETASK-6": {
                "title":
                self.task_group_tasks_historical[0]["title"] + " one",
                "description":
                self.task_group_tasks_historical[0]["description"] + " one",
                "start_date":
                "2014-04-01",
                "end_date":
                "2014-04-06",
                "finished_date":
                "2014-05-01 00:00:00",
                "verified_date":
                "2014-06-06 00:00:00",
                "status":
                "Assigned"
            },
            "CYCLETASK-7": {
                "title":
                self.task_group_tasks_historical[1]["title"] + " two",
                "description":
                self.task_group_tasks_historical[1]["description"] + " two",
                "start_date":
                "2014-04-07",
                "end_date":
                "2014-04-12",
                "finished_date":
                "2014-05-07 00:00:00",
                "verified_date":
                "None",
                "status":
                "Declined"
            },
            "CYCLETASK-8": {
                "title":
                self.task_group_tasks_historical[2]["title"] + " three",
                "description":
                self.task_group_tasks_historical[2]["description"] + " three",
                "start_date":
                "2014-04-13",
                "end_date":
                "2014-04-18",
                "finished_date":
                "2014-05-13 00:00:00",
                "verified_date":
                "2014-06-18 00:00:00",
                "status":
                "InProgress"
            },
            "CYCLETASK-9": {
                "title":
                self.task_group_tasks_historical[3]["title"] + " four",
                "description":
                self.task_group_tasks_historical[3]["description"] + " four",
                "start_date":
                "2014-04-19",
                "end_date":
                "2014-04-24",
                "finished_date":
                "2014-05-19 00:00:00",
                "verified_date":
                "None",
                "status":
                "Finished"
            },
            "CYCLETASK-10": {
                "title":
                self.task_group_tasks_historical[4]["title"] + " five",
                "description":
                self.task_group_tasks_historical[4]["description"] + " five",
                "start_date":
                "2014-04-25",
                "end_date":
                "2014-04-30",
                "finished_date":
                "2014-05-25 00:00:00",
                "verified_date":
                "2014-06-30 00:00:00",
                "status":
                "Verified"
            }
        }

        # Below is description of warning for non-importable columns. It is needed
        # for test_cycle_task_warnings.
        importable_column_names = []
        for field_name in CycleTaskGroupObjectTask.IMPORTABLE_FIELDS:
            if field_name == 'slug':
                continue
            # pylint: disable=protected-access
            importable_column_names.append(
                CycleTaskGroupObjectTask._aliases.get(field_name, field_name))
        self.expected_warnings = {
            'Cycle Task Group Object Task': {
                'block_warnings': {
                    errors.ONLY_IMPORTABLE_COLUMNS_WARNING.format(
                        line=2, columns=", ".join(importable_column_names))
                },
            }
        }

        # This is an error message which should be shown during
        # test_cycle_task_create_error test
        self.expected_create_error = {
            'Cycle Task Group Object Task': {
                'row_errors': {errors.CREATE_INSTANCE_ERROR.format(line=13)}
            }
        }

        # Below is expected date errors for test_cycle_task_date_error. They should
        # be shown during date validator's tests.
        self.expected_date_error = {
            'Cycle Task Group Object Task': {
                'row_errors': {
                    errors.INVALID_START_END_DATES.format(
                        line=3,
                        start_date="Start Date",
                        end_date="End Date",
                    ),
                    errors.INVALID_STATUS_DATE_CORRELATION.format(
                        line=4,
                        date="Actual Finish Date",
                        status="not Finished",
                    ),
                    errors.INVALID_STATUS_DATE_CORRELATION.format(
                        line=5,
                        date="Actual Verified Date",
                        status="not Verified",
                    ),
                    errors.INVALID_STATUS_DATE_CORRELATION.format(
                        line=6,
                        date="Actual Verified Date",
                        status="not Verified",
                    ),
                    errors.INVALID_START_END_DATES.format(
                        line=7,
                        start_date="Actual Finish Date",
                        end_date="Actual Verified Date",
                    ),
                    errors.INVALID_START_END_DATES.format(
                        line=8,
                        start_date="Start Date",
                        end_date="End Date",
                    ),
                    errors.MISSING_VALUE_ERROR.format(
                        line=9,
                        column_name="Actual Finish Date",
                    ),
                    errors.INVALID_START_END_DATES.format(
                        line=10,
                        start_date="Actual Finish Date",
                        end_date="Actual Verified Date",
                    ),
                },
            }
        }
        # Below is expected cycle-tasks data which should appear in test DB after
        # test_cycle_task_date_error run
        self.expected_cycle_task_date_error = dict()
        self.expected_cycle_task_date_error.update(
            self.generated_cycle_tasks_active)
        self.expected_cycle_task_date_error.update(
            self.generated_cycle_tasks_historical)
        self.expected_cycle_task_date_error["CYCLETASK-9"] = (
            self.expected_cycle_task_correct["CYCLETASK-9"])
        self.expected_cycle_task_date_error["CYCLETASK-10"] = (
            self.expected_cycle_task_correct["CYCLETASK-10"])

        # Expected error message which should be shown after
        # test_cycle_task_permission_error run
        self.expected_permission_error = {
            'Cycle Task Group Object Task': {
                'block_errors': {errors.PERMISSION_ERROR.format(line=2)}
            }
        }
class TestTaskDueNotifications(TestCase):
  """Test suite for task due soon/today notifications."""

  # pylint: disable=invalid-name

  def _fix_notification_init(self):
    """Fix Notification object init function.

    This is a fix needed for correct created_at field when using freezgun. By
    default the created_at field is left empty and filed by database, which
    uses system time and not the fake date set by freezugun plugin. This fix
    makes sure that object created in freeze_time block has all dates set with
    the correct date and time.
    """
    def init_decorator(init):
      """"Adjust the value of the object's created_at attribute to now."""
      @functools.wraps(init)
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    models.Notification.__init__ = init_decorator(models.Notification.__init__)

  def setUp(self):
    super(TestTaskDueNotifications, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    models.Notification.query.delete()

    self._fix_notification_init()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")

    person_dict = {
        "href": "/api/people/{}".format(self.user.id),
        "id": self.user.id,
        "type": "Person",
    }

    self.one_time_workflow = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        "is_verification_needed": False,
        "owners": [person_dict.copy()],
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict.copy(),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "contact": person_dict.copy(),
                "start_date": date(2017, 5, 15),
                "end_date": date(2017, 6, 11),
            }, {
                "title": "task 2",
                "description": "some task 2",
                "contact": person_dict.copy(),
                "start_date": date(2017, 5, 8),
                "end_date": date(2017, 6, 12),
            }, {
                "title": "task 3",
                "description": "some task 3",
                "contact": person_dict.copy(),
                "start_date": date(2017, 5, 31),
                "end_date": date(2017, 6, 13),
            }, {
                "title": "task 4",
                "description": "some task 4",
                "contact": person_dict.copy(),
                "start_date": date(2017, 6, 2),
                "end_date": date(2017, 6, 14),
            }, {
                "title": "task 5",
                "description": "some task 5",
                "contact": person_dict.copy(),
                "start_date": date(2017, 6, 8),
                "end_date": date(2017, 6, 15),
            }],
            "task_group_objects": self.random_objects
        }]
    }

  @ddt.unpack
  @ddt.data(
      ("2017-06-12 12:12:12", ["task 1"], ["task 2"], ["task 3"]),
      ("2017-06-13 13:13:13", ["task 1", "task 2"], ["task 3"], ["task 4"]),
  )
  @patch("ggrc.notifications.common.send_email")
  def test_creating_obsolete_notifications(
      self, fake_now, expected_overdue, expected_due_today, expected_due_in, _
  ):
    """Notifications already obsolete on creation date should not be created.
    """
    with freeze_time("2017-06-12 09:39:32"):
      tmp = self.one_time_workflow.copy()
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    user = models.Person.query.get(self.user.id)

    with freeze_time(fake_now):
      # mark all yeasterday notifications as sent
      models.all_models.Notification.query.filter(
          sa.func.DATE(models.all_models.Notification.send_on) < date.today()
      ).update({models.all_models.Notification.sent_at:
                datetime.now() - timedelta(1)},
               synchronize_session="fetch")

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})

      actual_overdue = [n['title'] for n in
                        user_notifs.get("task_overdue", {}).itervalues()]
      actual_overdue.sort()
      self.assertEqual(actual_overdue, expected_overdue)

      self.assertEqual(
          [n['title'] for n in user_notifs.get("due_today", {}).itervalues()],
          expected_due_today)

      self.assertEqual(
          [n['title'] for n in user_notifs.get("due_in", {}).itervalues()],
          expected_due_in)
Exemple #37
0
class TestOneTimeWorkflowNotification(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestOneTimeWorkflowNotification, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()

    self.random_objects = self.object_generator.generate_random_objects()
    self.random_people = self.object_generator.generate_random_people(
        user_role="Administrator"
    )
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  def test_one_time_wf_activate(self):
    def get_person(person_id):
      return db.session.query(Person).filter(Person.id == person_id).one()
    with freeze_time("2015-04-10"):
      _, wf = self.wf_generator.generate_workflow(self.one_time_workflow_1)

      _, cycle = self.wf_generator.generate_cycle(wf)
      self.wf_generator.activate_workflow(wf)

      person_2 = get_person(self.random_people[2].id)

    with freeze_time("2015-04-11"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(person_2.email, notif_data)
      self.assertIn("cycle_started", notif_data[person_2.email])
      self.assertIn(cycle.id, notif_data[person_2.email]["cycle_started"])
      self.assertIn("my_tasks",
                    notif_data[person_2.email]["cycle_data"][cycle.id])

      person_1 = get_person(self.random_people[0].id)

    with freeze_time("2015-05-03"):  # two days befor due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(person_1.email, notif_data)
      self.assertNotIn("due_in", notif_data[person_1.email])
      self.assertNotIn("due_today", notif_data[person_1.email])

    with freeze_time("2015-05-04"):  # one day befor due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(len(notif_data[person_1.email]["due_in"]), 1)

    with freeze_time("2015-05-05"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(len(notif_data[person_1.email]["due_today"]), 1)

  @patch("ggrc.notifications.common.send_email")
  def test_one_time_wf_activate_single_person(self, mock_mail):

    with freeze_time("2015-04-10"):
      user = "******"
      _, wf = self.wf_generator.generate_workflow(
          self.one_time_workflow_single_person)

      _, cycle = self.wf_generator.generate_cycle(wf)
      self.wf_generator.activate_workflow(wf)

    with freeze_time("2015-04-11"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn("cycle_started", notif_data[user])
      self.assertIn(cycle.id, notif_data[user]["cycle_started"])
      self.assertIn("my_tasks", notif_data[user]["cycle_data"][cycle.id])
      self.assertIn("cycle_tasks", notif_data[user]["cycle_data"][cycle.id])
      self.assertIn(
          "my_task_groups", notif_data[user]["cycle_data"][cycle.id])
      self.assertIn("cycle_url", notif_data[user]["cycle_started"][cycle.id])

      cycle = Cycle.query.get(cycle.id)
      cycle_data = notif_data[user]["cycle_data"][cycle.id]
      for task in cycle.cycle_task_group_object_tasks:
        self.assertIn(task.id, cycle_data["my_tasks"])
        self.assertIn(task.id, cycle_data["cycle_tasks"])
        self.assertIn("title", cycle_data["my_tasks"][task.id])
        self.assertIn("title", cycle_data["cycle_tasks"][task.id])
        self.assertIn("cycle_task_url", cycle_data["cycle_tasks"][task.id])

    with freeze_time("2015-05-03"):  # two days before due date
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user, notif_data)
      self.assertNotIn("due_in", notif_data[user])
      self.assertNotIn("due_today", notif_data[user])

    with freeze_time("2015-05-04"):  # one day before due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(len(notif_data[user]["due_in"]), 2)

    with freeze_time("2015-05-05"):  # due date
      _, notif_data = common.get_daily_notifications()
      self.assertEqual(len(notif_data[user]["due_today"]), 2)

      common.send_daily_digest_notifications()
      self.assertEqual(mock_mail.call_count, 1)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    role_id = all_models.AccessControlRole.query.filter(
        all_models.AccessControlRole.name == "Task Assignees",
        all_models.AccessControlRole.object_type == "TaskGroupTask",
    ).one().id
    self.one_time_workflow_1 = {
        "title": "one time test workflow",
        "description": "some test workflow",
        "notify_on_change": True,
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.random_people[2].id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.random_people[0].id)
                ],
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task 2",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.random_people[1].id)
                ],
                "start_date": date(2015, 5, 4),
                "end_date": date(2015, 5, 7),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "contact": person_dict(self.random_people[2].id),
            "task_group_tasks": [{
                "title": "task 1 in tg 2",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.random_people[0].id)
                ],
                "start_date": date(2015, 5, 8),  # friday
                "end_date": date(2015, 5, 12),
            }, {
                "title": "task 2 in tg 2",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.random_people[2].id)
                ],
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": []
        }]
    }

    user = Person.query.filter(Person.email == "*****@*****.**").one().id

    self.one_time_workflow_single_person = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(user),
            "task_group_tasks": [{
                "title": u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                "description": "some task. ",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, user)
                ],
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }, {
                "title": "task 2",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, user)
                ],
                "start_date": date(2015, 5, 4),
                "end_date": date(2015, 5, 7),
            }],
            "task_group_objects": self.random_objects[:2]
        }, {
            "title": "another one time task group",
            "contact": person_dict(user),
            "task_group_tasks": [{
                "title": u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, user)
                ],
                "start_date": date(2015, 5, 8),  # friday
                "end_date": date(2015, 5, 12),
            }, {
                "title": "task 2 in tg 2",
                "description": "some task",
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, user)
                ],
                "start_date": date(2015, 5, 1),  # friday
                "end_date": date(2015, 5, 5),
            }],
            "task_group_objects": []
        }]
    }
class TestTaskOverdueNotificationsUsingAPI(TestTaskOverdueNotifications):
  """Tests for overdue notifications when changing Tasks with an API."""

  # pylint: disable=invalid-name

  def setUp(self):
    super(TestTaskOverdueNotificationsUsingAPI, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    models.Notification.query.delete()

    self._fix_notification_init()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self._create_test_cases()

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_sending_overdue_notifications_for_tasks(self, is_vf_needed, _):
    """Overdue notifications should be sent for overdue tasks every day.

    Even if an overdue notification has already been sent, it should still be
    sent in every following daily digest f a task is still overdue.
    """
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    tasks = workflow.cycles[0].cycle_task_group_object_tasks
    task1_id = tasks[0].id
    task2_id = tasks[1].id

    user = models.Person.query.get(self.user.id)

    with freeze_time("2017-05-14 08:09:10"):
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

    with freeze_time("2017-05-15 08:09:10"):  # task 1 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

    with freeze_time("2017-05-16 08:09:10"):  # task 2 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id])

    with freeze_time("2017-05-17 08:09:10"):  # after both tasks' due dates
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id, task2_id])

      common.send_daily_digest_notifications()

    # even after sending the overdue notifications, they are sent again the
    # day after, too
    with freeze_time("2017-05-18 08:09:10"):
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)

      overdue_task_ids = sorted(user_notifs["task_overdue"].keys())
      self.assertEqual(overdue_task_ids, [task1_id, task2_id])

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_adjust_overdue_notifications_on_task_due_date_change(self,
                                                                is_vf_needed,
                                                                _):
    """Sending overdue notifications should adjust to task due date changes."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

      tasks = workflow.cycles[0].cycle_task_group_object_tasks
      task1, task2 = tasks
      self.wf_generator.modify_object(task2, {"end_date": date(2099, 12, 31)})

      user = models.Person.query.get(self.user.id)

    with freeze_time("2017-05-16 08:09:10"):  # a day after task1 due date
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

      # change task1 due date, there should be no overdue notification anymore
      self.wf_generator.modify_object(task1, {"end_date": date(2017, 5, 16)})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

      # change task1 due date to the past there should a notification again
      self.wf_generator.modify_object(task1, {"end_date": date(2017, 5, 14)})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_adjust_overdue_notifications_on_task_status_change(self,
                                                              is_vf_needed,
                                                              _):
    """Sending overdue notifications should take task status into account."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

      tasks = workflow.cycles[0].cycle_task_group_object_tasks
      task1, task2 = tasks
      self.wf_generator.modify_object(task2, {"end_date": date(2099, 12, 31)})

      user = models.Person.query.get(self.user.id)
      user_email = user.email
    if is_vf_needed:
      non_final_states = [CycleTaskGroupObjectTask.ASSIGNED,
                          CycleTaskGroupObjectTask.IN_PROGRESS,
                          CycleTaskGroupObjectTask.FINISHED,
                          CycleTaskGroupObjectTask.DECLINED]
      final_state = CycleTaskGroupObjectTask.VERIFIED
    else:
      non_final_states = [CycleTaskGroupObjectTask.ASSIGNED,
                          CycleTaskGroupObjectTask.IN_PROGRESS]
      final_state = CycleTaskGroupObjectTask.FINISHED

    with freeze_time("2017-05-16 08:09:10"):  # a day after task1 due date
      for state in non_final_states:
        # clear all notifications before before changing the task status
        models.Notification.query.delete()
        _, notif_data = common.get_daily_notifications()
        self.assertEqual(notif_data, {})

        self.wf_generator.modify_object(task1, {"status": state})

        _, notif_data = common.get_daily_notifications()
        user_notifs = notif_data.get(user_email, {})
        self.assertIn("task_overdue", user_notifs)
        self.assertEqual(len(user_notifs["task_overdue"]), 1)

      # WITHOUT clearing the overdue notifications, move the task to "verified"
      # state, and the overdue notification should disappear.

      self.wf_generator.modify_object(task1, {"status": final_state})
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertNotIn("task_overdue", user_notifs)

  @ddt.data(True, False)
  @patch("ggrc.notifications.common.send_email")
  def test_stop_sending_overdue_notification_if_task_gets_deleted(self,
                                                                  is_vf_needed,
                                                                  _):
    """Overdue notifications should not be sent for deleted tasks."""
    with freeze_time("2017-05-15 14:25:36"):
      tmp = self.one_time_workflow.copy()
      tmp['is_verification_needed'] = is_vf_needed
      _, workflow = self.wf_generator.generate_workflow(tmp)
      self.wf_generator.generate_cycle(workflow)
      response, workflow = self.wf_generator.activate_workflow(workflow)
      self.assert200(response)

    tasks = workflow.cycles[0].cycle_task_group_object_tasks
    task1, task2 = tasks

    user = models.Person.query.get(self.user.id)
    user_email = user.email

    with freeze_time("2017-10-16 08:09:10"):  # long after both task due dates
      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 2)

      db.session.delete(task2)
      db.session.commit()

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user_email, {})
      self.assertIn("task_overdue", user_notifs)
      self.assertEqual(len(user_notifs["task_overdue"]), 1)

      db.session.delete(task1)
      db.session.commit()

      _, notif_data = common.get_daily_notifications()
      user_notifs = notif_data.get(user.email, {})
      self.assertNotIn("task_overdue", user_notifs)

  def _create_test_cases(self):
    """Create configuration to use for generating a new workflow."""
    def person_dict(person_id):
      return {
          "href": "/api/people/" + str(person_id),
          "id": person_id,
          "type": "Person"
      }
    role_id = models.all_models.AccessControlRole.query.filter(
        models.all_models.AccessControlRole.name == "Task Assignees",
        models.all_models.AccessControlRole.object_type == "TaskGroupTask",
    ).one().id
    self.one_time_workflow = {
        "title": "one time test workflow",
        "notify_on_change": True,
        "description": "some test workflow",
        # admin will be current user with id == 1
        "task_groups": [{
            "title": "one time task group",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "title": "task 1",
                "description": "some task",
                "start_date": date(2017, 5, 5),  # Friday
                "end_date": date(2017, 5, 15),
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
            }, {
                "title": "task 2",
                "description": "some task 2",
                "start_date": date(2017, 5, 5),  # Friday
                "end_date": date(2017, 5, 16),
                "access_control_list": [
                    acl_helper.get_acl_json(role_id, self.user.id)],
            }],
            "task_group_objects": self.random_objects
        }]
    }
class TestNotificationsForDeletedObjects(TestCase):

  """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """

  def setUp(self):
    super(TestNotificationsForDeletedObjects, self).setUp()
    self.api = Api()
    self.wf_generator = WorkflowsGenerator()
    self.object_generator = ObjectGenerator()
    Notification.query.delete()

    self.random_objects = self.object_generator.generate_random_objects(2)
    _, self.user = self.object_generator.generate_person(
        user_role="Administrator")
    self.create_test_cases()

    def init_decorator(init):
      def new_init(self, *args, **kwargs):
        init(self, *args, **kwargs)
        if hasattr(self, "created_at"):
          self.created_at = datetime.now()
      return new_init

    Notification.__init__ = init_decorator(Notification.__init__)

  @patch("ggrc.notifications.common.send_email")
  def test_delete_activated_workflow(self, mock_mail):

    with freeze_time("2015-02-01 13:39:20"):
      _, workflow = self.wf_generator.generate_workflow(self.quarterly_wf_1)
      response, workflow = self.wf_generator.activate_workflow(workflow)

      self.assert200(response)

      user = Person.query.get(self.user.id)

    with freeze_time("2015-01-01 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertNotIn(user.email, notif_data)

    with freeze_time("2015-01-29 13:39:20"):
      _, notif_data = common.get_daily_notifications()
      self.assertIn(user.email, notif_data)
      self.assertIn("cycle_starts_in", notif_data[user.email])

      workflow = Workflow.query.get(workflow.id)
      # After workflow deletion its notifications object_ids updated to 0
      # value, this is the error, them should be deleted
      # so this query checks existence of notifications with object_id
      # equal to workflow id or 0 id before and
      # after deletion workflow instance
      exists_qs = db.session.query(
          Notification.query.filter(
              Notification.object_type == workflow.__class__.__name__,
              Notification.object_id.in_((workflow.id, 0))
          ).exists()
      )
      self.assertTrue(exists_qs.one()[0])
      response = self.wf_generator.api.delete(workflow)
      self.assert200(response)

      self.assertFalse(exists_qs.one()[0])

      _, notif_data = common.get_daily_notifications()
      user = Person.query.get(self.user.id)

      self.assertNotIn(user.email, notif_data)

  def create_test_cases(self):
    def person_dict(person_id):
      return {
          "href": "/api/people/%d" % person_id,
          "id": person_id,
          "type": "Person"
      }

    self.quarterly_wf_1 = {
        "title": "quarterly wf 1",
        "notify_on_change": True,
        "description": "",
        "owners": [person_dict(self.user.id)],
        "frequency": "quarterly",
        "task_groups": [{
            "title": "tg_1",
            "contact": person_dict(self.user.id),
            "task_group_tasks": [{
                "contact": person_dict(self.user.id),
                "description": factories.random_str(100),
                "relative_start_day": 5,
                "relative_start_month": 2,
                "relative_end_day": 25,
                "relative_end_month": 2,
            },
            ],
        },
        ]
    }
class TestOneTimeWorkflowNotification(TestCase):
    """ This class contains simple one time workflow tests that are not
  in the gsheet test grid
  """
    def setUp(self):
        super(TestOneTimeWorkflowNotification, self).setUp()
        self.api = Api()
        self.wf_generator = WorkflowsGenerator()
        self.object_generator = ObjectGenerator()

        self.random_objects = self.object_generator.generate_random_objects()
        self.person_1 = self.create_user_with_role(role="Administrator")
        self.person_2 = self.create_user_with_role(role="Administrator")
        self.secondary_assignee = self.create_user_with_role(role="Reader")
        self.create_test_cases()

        def init_decorator(init):
            def new_init(self, *args, **kwargs):
                init(self, *args, **kwargs)
                if hasattr(self, "created_at"):
                    self.created_at = datetime.now()

            return new_init

        all_models.Notification.__init__ = init_decorator(
            all_models.Notification.__init__)

    def test_one_time_wf_activate(self):
        """Test one time wf activation"""

        with freeze_time("2015-04-10"):
            _, wf = self.wf_generator.generate_workflow(
                self.one_time_workflow_1)
            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

            person_1, person_2 = self.person_1, self.person_2
            secondary_assignee = self.secondary_assignee

        task_assignees = [person_1, person_2, secondary_assignee]
        with freeze_time("2015-04-11"):
            _, notif_data = common.get_daily_notifications()

            self.assertIn(person_2.email, notif_data)

            # cycle started notifs available only for contact
            self.assertIn("cycle_started", notif_data[person_2.email])
            self.assertIn(cycle.id,
                          notif_data[person_2.email]["cycle_started"])

            for user in task_assignees:
                self.assertIn("my_tasks",
                              notif_data[user.email]["cycle_data"][cycle.id])

        with freeze_time("2015-05-03"):  # two days before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertIn(user.email, notif_data)
                self.assertNotIn("due_in", notif_data[user.email])
                self.assertNotIn("due_today", notif_data[user.email])

        with freeze_time("2015-05-04"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertEqual(len(notif_data[user.email]["due_in"]), 1)

        with freeze_time("2015-05-05"):  # due date
            _, notif_data = common.get_daily_notifications()
            for user in task_assignees:
                self.assertEqual(len(notif_data[user.email]["due_today"]), 1)

        with freeze_time("2015-05-10"):
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(
                len(notif_data[secondary_assignee.email]["due_in"]), 2)

    @patch("ggrc.notifications.common.send_email")
    def test_one_time_wf_activate_single_person(self, mock_mail):
        """Test one time wf activation with single person"""

        with freeze_time("2015-04-10"):
            user = "******"
            _, wf = self.wf_generator.generate_workflow(
                self.one_time_workflow_single_person)

            _, cycle = self.wf_generator.generate_cycle(wf)
            self.wf_generator.activate_workflow(wf)

        with freeze_time("2015-04-11"):
            _, notif_data = common.get_daily_notifications()
            self.assertIn("cycle_started", notif_data[user])
            self.assertIn(cycle.id, notif_data[user]["cycle_started"])
            self.assertIn("my_tasks", notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("cycle_tasks",
                          notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("my_task_groups",
                          notif_data[user]["cycle_data"][cycle.id])
            self.assertIn("cycle_url",
                          notif_data[user]["cycle_started"][cycle.id])

            cycle = Cycle.query.get(cycle.id)
            cycle_data = notif_data[user]["cycle_data"][cycle.id]
            for task in cycle.cycle_task_group_object_tasks:
                self.assertIn(task.id, cycle_data["my_tasks"])
                self.assertIn(task.id, cycle_data["cycle_tasks"])
                self.assertIn("title", cycle_data["my_tasks"][task.id])
                self.assertIn("title", cycle_data["cycle_tasks"][task.id])
                self.assertIn("cycle_task_url",
                              cycle_data["cycle_tasks"][task.id])

        with freeze_time("2015-05-03"):  # two days before due date
            _, notif_data = common.get_daily_notifications()
            self.assertIn(user, notif_data)
            self.assertNotIn("due_in", notif_data[user])
            self.assertNotIn("due_today", notif_data[user])

        with freeze_time("2015-05-04"):  # one day before due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[user]["due_in"]), 2)

        with freeze_time("2015-05-05"):  # due date
            _, notif_data = common.get_daily_notifications()
            self.assertEqual(len(notif_data[user]["due_today"]), 2)

            common.send_daily_digest_notifications()
            self.assertEqual(mock_mail.call_count, 1)

    def create_test_cases(self):
        def person_dict(person_id):
            return {
                "href": "/api/people/%d" % person_id,
                "id": person_id,
                "type": "Person"
            }

        task_assignee_role_id = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id

        secondary_task_assignee_id = all_models.AccessControlRole.query.filter(
            all_models.AccessControlRole.name == "Task Secondary Assignees",
            all_models.AccessControlRole.object_type == "TaskGroupTask",
        ).one().id
        self.one_time_workflow_1 = {
            "title":
            "one time test workflow",
            "description":
            "some test workflow",
            "notify_on_change":
            True,
            # admin will be current user with id == 1
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "contact":
                    person_dict(self.person_2.id),
                    "task_group_tasks": [
                        {
                            "title":
                            "task 1",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, self.person_1.id),
                                acl_helper.get_acl_json(
                                    secondary_task_assignee_id,
                                    self.secondary_assignee.id),
                            ],
                            "start_date":
                            date(2015, 5, 1),  # friday
                            "end_date":
                            date(2015, 5, 5),
                        },
                        {
                            "title":
                            "task 2",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, self.person_1.id),
                                acl_helper.get_acl_json(
                                    secondary_task_assignee_id,
                                    self.secondary_assignee.id),
                            ],
                            "start_date":
                            date(2015, 5, 4),
                            "end_date":
                            date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "contact":
                    person_dict(self.person_2.id),
                    "task_group_tasks": [
                        {
                            "title":
                            "task 1 in tg 2",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, self.person_1.id)
                            ],
                            "start_date":
                            date(2015, 5, 8),  # friday
                            "end_date":
                            date(2015, 5, 12),
                        },
                        {
                            "title":
                            "task 2 in tg 2",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, self.person_2.id)
                            ],
                            "start_date":
                            date(2015, 5, 1),  # friday
                            "end_date":
                            date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }

        user = all_models.Person.query.filter(
            all_models.Person.email == "*****@*****.**").one().id

        self.one_time_workflow_single_person = {
            "title":
            "one time test workflow",
            "notify_on_change":
            True,
            "description":
            "some test workflow",
            # admin will be current user with id == 1
            "task_groups": [
                {
                    "title":
                    "one time task group",
                    "contact":
                    person_dict(user),
                    "task_group_tasks": [
                        {
                            "title":
                            u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                            "description":
                            "some task. ",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, user)
                            ],
                            "start_date":
                            date(2015, 5, 1),  # friday
                            "end_date":
                            date(2015, 5, 5),
                        },
                        {
                            "title":
                            "task 2",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, user)
                            ],
                            "start_date":
                            date(2015, 5, 4),
                            "end_date":
                            date(2015, 5, 7),
                        }
                    ],
                    "task_group_objects":
                    self.random_objects[:2]
                },
                {
                    "title":
                    "another one time task group",
                    "contact":
                    person_dict(user),
                    "task_group_tasks": [
                        {
                            "title":
                            u"task 1 \u2062 WITH AN UMBRELLA ELLA ELLA. \u2062",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, user)
                            ],
                            "start_date":
                            date(2015, 5, 8),  # friday
                            "end_date":
                            date(2015, 5, 12),
                        },
                        {
                            "title":
                            "task 2 in tg 2",
                            "description":
                            "some task",
                            "access_control_list": [
                                acl_helper.get_acl_json(
                                    task_assignee_role_id, user)
                            ],
                            "start_date":
                            date(2015, 5, 1),  # friday
                            "end_date":
                            date(2015, 5, 5),
                        }
                    ],
                    "task_group_objects": []
                }
            ]
        }