This algorithm is a legacy one. The API has changed since its implementation. New versions and forks will need to be updated.
This algorithm is an analyzer. It can only be used on analysis blocks.

Algorithms have at least one input and one output. All algorithm endpoints are organized in groups. Groups are used by the platform to indicate which inputs and outputs are synchronized together. The first group is automatically synchronized with the channel defined by the block in which the algorithm is deployed.

Group: test

Endpoint Name Data Format Nature
scores_test system/array_1d_floats/1 Input
label_test system/text/1 Input

Group: dev

Endpoint Name Data Format Nature
scores_dev system/array_1d_floats/1 Input
label_dev system/text/1 Input

Analyzers may produce any number of results. Once experiments using this analyzer are done, you may display the results or filter experiments using criteria based on them.

Name Type
test_frr float32
dev_eer float32
dev_roc plot/isoroc/1
test_roc plot/isoroc/1
test_numNegatives int32
dev_numNegatives int32
dev_scoreDistribution plot/bar/1
test_numPositives int32
dev_far float32
test_hter float32
dev_numPositives int32
test_scoreDistribution plot/bar/1
test_far float32
dev_frr float32
dev_eer_threshold float32
xxxxxxxxxx
131
 
1
import numpy
2
import bob.measure
3
4
# same functionality as earlier version but using Bob_v2
5
6
class Algorithm:
7
8
    def __init__(self):
9
        self.positives_dev = None
10
        self.negatives_dev = None
11
        self.positives_test = []
12
        self.negatives_test = []
13
        self.n_bins = 50
14
15
    def process(self, inputs, outputs):
16
17
        # accumulate the dev scores
18
        if self.positives_dev is None:
19
            self.positives_dev = []
20
            self.negatives_dev = []
21
22
            group = inputs.groupOf('scores_dev')
23
            while group.hasMoreData():
24
              group.next()
25
              data_dev = inputs['scores_dev'].data.value
26
              label_dev = inputs['label_dev'].data.text
27
              if label_dev == 'real':
28
                self.positives_dev.append(data_dev)
29
              else:  
30
                self.negatives_dev.append(data_dev)
31
          
32
33
        # accumulate the test scores
34
        data_test = inputs['scores_test'].data.value
35
        label_test = inputs['label_test'].data.text
36
        if label_test == 'real':
37
            self.positives_test.append(data_test)
38
        else:  
39
            self.negatives_test.append(data_test)
40
41
        # once all values are received, evaluate the scores
42
        if not(inputs.hasMoreData()):
43
            self.positives_dev = numpy.vstack(self.positives_dev).flatten()
44
            self.negatives_dev = numpy.vstack(self.negatives_dev).flatten()
45
            
46
            self.positives_test = numpy.vstack(self.positives_test).flatten()
47
            self.negatives_test = numpy.vstack(self.negatives_test).flatten()
48
            
49
            dev_hist_pos, dev_bin_pos = numpy.histogram(self.positives_dev, self.n_bins)
50
            dev_hist_neg, dev_bin_neg = numpy.histogram(self.negatives_dev, self.n_bins)
51
            
52
            test_hist_pos, test_bin_pos = numpy.histogram(self.positives_test, self.n_bins)
53
            test_hist_neg, test_bin_neg = numpy.histogram(self.negatives_test, self.n_bins)
54
            
55
            eer_threshold = bob.measure.eer_threshold(self.negatives_dev, self.positives_dev)
56
            far_dev, frr_dev   = bob.measure.farfrr(self.negatives_dev, self.positives_dev, eer_threshold)
57
            far_test, frr_test = bob.measure.farfrr(self.negatives_test, self.positives_test, eer_threshold)
58
            roc_points_dev     = bob.measure.roc(self.negatives_dev, self.positives_dev, 100)
59
            roc_points_test    = bob.measure.roc(self.negatives_test, self.positives_test, 100)
60
61
            # writes the output back to the platform
62
            outputs.write({
63
                'dev_eer': numpy.float32((far_dev+frr_dev) / 2.),
64
                'test_hter': numpy.float32((far_test+frr_test) / 2.),
65
                'dev_far': numpy.float32(far_dev),
66
                'dev_frr': numpy.float32(frr_dev),
67
                'test_far': numpy.float32(far_test),
68
                'test_frr': numpy.float32(frr_test),
69
                'dev_numPositives': numpy.int32(len(self.positives_dev)),
70
                'dev_numNegatives': numpy.int32(len(self.negatives_dev)),
71
                'test_numPositives': numpy.int32(len(self.positives_test)),
72
                'test_numNegatives': numpy.int32(len(self.negatives_test)),
73
                'dev_eer_threshold': numpy.float32(eer_threshold),
74
                'dev_roc': {
75
                        "data":
76
                        [   
77
                            {                              
78
                                "label": "roc_dev",
79
                                "false_positives": roc_points_dev[1],
80
                                "false_negatives": roc_points_dev[0],
81
                                "number_of_positives": numpy.uint64(len(self.positives_dev)),
82
                                "number_of_negatives": numpy.uint64(len(self.negatives_dev)),
83
                            }   
84
                        ]   
85
                    },
86
                'test_roc': {
87
                    "data":
88
                        [   
89
                            {   
90
                                "label": "roc_test",
91
                                "false_positives": roc_points_test[1],
92
                                "false_negatives": roc_points_test[0],
93
                                "number_of_positives": numpy.uint64(len(self.positives_test)),
94
                                "number_of_negatives": numpy.uint64(len(self.negatives_test)),
95
                            }   
96
                        ]   
97
                    },
98
                
99
                'dev_scoreDistribution': {
100
                    "data":
101
                        [
102
                            {
103
                                "label": "negative scores",
104
                                "x": dev_bin_neg[:-1],
105
                                "y": dev_hist_neg,
106
                            },
107
                            {
108
                                "label": "positive scores",
109
                                "x": dev_bin_pos[:-1],
110
                                "y": dev_hist_pos,
111
                            }
112
                        ]
113
                    },
114
                'test_scoreDistribution': {
115
                    "data":
116
                        [
117
                            {
118
                                "label": "negative scores",
119
                                "x": test_bin_neg[:-1],
120
                                "y": test_hist_neg,
121
                            },
122
                            {
123
                                "label": "positive scores",
124
                                "x": test_bin_pos[:-1],
125
                                "y": test_hist_pos,
126
                            }
127
                        ]
128
                    },
129
            })
130
        return True
131

The code for this algorithm in Python
The ruler at 80 columns indicate suggested POSIX line breaks (for readability).
The editor will automatically enlarge to accomodate the entirety of your input
Use keyboard shortcuts for search/replace and faster editing. For example, use Ctrl-F (PC) or Cmd-F (Mac) to search through this box

An algorithm that implements standard metrics for antispoofing evaluation.

Specifically, it returns:

  • eer: the equal error rate (EER) on the development set.
  • hter: the half total error rate (HTER) on the test set
  • far_dev: the false acceptance rate (FAR) on the development set
  • frr_dev: the false rejection rate (FRR) on the development set
  • far_test: the false acceptance rate (FAR) on the test set
  • frr_test: the false rejection rate (FRR) on the test set
  • number_of_positives_dev: the number of positive trials on the development set
  • number_of_negatives_dev: the number of negative trials on the development set
  • number_of_positives_test: the number of positive trials on the test set
  • number_of_negatives_test: the number of negative trials on the test set
  • threshold: the threshold at the equal error rate on the development set
  • roc_dev: the receiver operating characteristic (ROC) curve on the development set
  • roc_test: the receiver operating characteristic (ROC) curve on the test set

This implementation relies on the 'measure' package from the Bob library. See http://www.idiap.ch/software/bob/docs/releases/last/sphinx/html/measure/ for more details.

Experiments

Updated Name Databases/Protocols Analyzers
smarcel/ivana7c/simple-antispoofing-updated/1/replay2-antispoofing-lbp-histograms-fix replay/3@grandtest Kanma/iqm_spoof_eer_analyzer/1
Created with Raphaël 2.1.2[compare]sbhatta/iqm_spoof_eer_analyzer/9anjos/antispoofing_analyzer/12015Nov19Kanma/iqm_spoof_eer_analyzer/12016Jan152017Dec4

This table shows the number of times this algorithm has been successfully run using the given environment. Note this does not provide sufficient information to evaluate if the algorithm will run when submitted to different conditions.

Terms of Service | Contact Information | BEAT platform version 2.2.1b0 | © Idiap Research Institute - 2013-2025