Beispiel #1
0
    def test_print_stats_per_epoch_is_false(self, mock_stderr):
        # To test if a method prints stats in TQDM re-enable TQDM printing
        constants.TQDM_DISABLE = False
        reload(training)

        sys.stderr = sys.__stderr__
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)
        initial_model = deepcopy(model)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 1
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        self.assertEqual(mock_stderr.getvalue().count("Train_loss"), 0)
        self.assertEqual(mock_stderr.getvalue().count("Train_acc"), 0)
        self.assertEqual(mock_stderr.getvalue().count("Val_acc"), 0)

        # At the end of the method testing, disable TQDM printing again
        constants.TQDM_DISABLE = True
        reload(training)
Beispiel #2
0
    def test_error_improves_with_more_than_one_class_in_the_training_set(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        # Even with classes randomly assigned, the model should be able to overfit to the training
        # set even if this doesn't imply improvement in the validation
        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        self.assertNotEqual(losses[0], min(losses))
        # Accuracy in the training set should be better than random
        self.assertGreaterEqual(max(accuracies[0]), 1. / n_classes)
Beispiel #3
0
    def test_error_improves_with_only_one_class_in_the_training_set(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)

        # Setting a small learning rate to ensure that improvement happens over the span
        # of multiple epochs
        optimizer = Adam(model.parameters(), lr=1e-8)
        error = CrossEntropyLoss()

        # Setting the number of training classes to 1, overfitting is easy and error should
        # necessarily improve
        train_loader = self.create_loader(n_features, 1)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        self.assertNotEqual(losses[0], min(losses))
Beispiel #4
0
    def test_returned_statistics_shape_is_correct_for_multibranch_model(self):
        n_features = 4
        n_classes = 3
        model = self.create_multibranch_model(n_features, n_classes)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features,
                                          n_classes,
                                          multibranch=True)
        val_loader = self.create_loader(n_features,
                                        n_classes,
                                        multibranch=True)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model,
            optimizer,
            error,
            train_loader,
            val_loader,
            num_epochs,
            print_stats_per_epoch,
            seq_start,
            seq_end,
            seq_step,
            patience,
            multibranch=True)

        self.assertEqual(losses.shape[1], 2)
        self.assertEqual(len(losses[:, 0]), num_epochs)
        self.assertEqual(len(losses[:, 1]), num_epochs)

        self.assertEqual(len(accuracies[0][:, 0]), num_epochs)
        self.assertEqual(len(accuracies[0][:, 1]), num_epochs)
        self.assertEqual(len(accuracies[1][:, 0]), num_epochs)
        self.assertEqual(len(accuracies[1][:, 1]), num_epochs)
Beispiel #5
0
    def test_model_with_best_val_accuracy_is_returned_for_multibranch_model(
            self):
        n_features = 4
        n_classes = 3
        model = self.create_multibranch_model(n_features, n_classes)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features,
                                          n_classes,
                                          multibranch=True)
        val_loader = self.create_loader(n_features,
                                        n_classes,
                                        multibranch=True)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model,
            optimizer,
            error,
            train_loader,
            val_loader,
            num_epochs,
            print_stats_per_epoch,
            seq_start,
            seq_end,
            seq_step,
            patience,
            multibranch=True)

        best_first_accuracy = training.evaluate(best_model[0], val_loader)
        self.assertEqual(max(np.array(accuracies[1])[:, 0]),
                         best_first_accuracy[0])
        best_second_accuracy = training.evaluate(best_model[1], val_loader)
        self.assertEqual(max(np.array(accuracies[1])[:, 1]),
                         best_second_accuracy[1])
Beispiel #6
0
    def test_positive_patience(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)
        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = 2

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)
        max_epochs_without_improvement = self.get_max_epochs_without_improvement(
            accuracies[1])
        # The training stops when the number of epochs without improvement surpasses the patience
        self.assertGreaterEqual(patience + 1, max_epochs_without_improvement)
Beispiel #7
0
    def test_zero_num_epochs(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)
        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 0
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        self.assertEqual(len(losses), num_epochs)
        self.assertEqual(len(accuracies[0]), num_epochs)
        self.assertEqual(len(accuracies[1]), num_epochs)
Beispiel #8
0
    def test_model_with_best_val_accuracy_is_returned(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)
        initial_model = deepcopy(model)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        best_model_accuracy = training.evaluate(best_model, val_loader)
        self.assertEqual(max(accuracies[1]), best_model_accuracy)
Beispiel #9
0
    def test_model_is_fitted(self):
        n_features = 4
        n_classes = 3
        model = self.create_model(n_features, n_classes)
        initial_model = deepcopy(model)

        optimizer = Adam(model.parameters())
        error = CrossEntropyLoss()

        train_loader = self.create_loader(n_features, n_classes)
        val_loader = self.create_loader(n_features, n_classes)
        num_epochs = 10
        print_stats_per_epoch = False

        seq_start, seq_end, seq_step = None, None, None
        patience = np.inf

        losses, accuracies, best_model = training.training_loop(
            model, optimizer, error, train_loader, val_loader, num_epochs,
            print_stats_per_epoch, seq_start, seq_end, seq_step, patience)

        # Assumes the accuracy of the model will improve
        for p1, p2 in zip(initial_model.parameters(), best_model.parameters()):
            self.assertFalse(torch.eq(p1.data, p2.data).any())