summaryrefslogtreecommitdiff
path: root/macaroonbakery/tests/test_oven.py
blob: 2976e940b57ee6067e0d054aaadbc03a2bf0cd81 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
# Copyright 2017 Canonical Ltd.
# Licensed under the LGPLv3, see LICENCE file for details.
from unittest import TestCase

import copy
from datetime import datetime, timedelta

import macaroonbakery

EPOCH = datetime(1900, 11, 17, 19, 00, 13, 0, None)
AGES = EPOCH + timedelta(days=10)


class TestOven(TestCase):
    def test_canonical_ops(self):
        canonical_ops_tests = (
            ('empty array', [], []),
            ('one element', [macaroonbakery.Op('a', 'a')],
             [macaroonbakery.Op('a', 'a')]),
            ('all in order',
             [macaroonbakery.Op('a', 'a'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('c', 'c')],
             [macaroonbakery.Op('a', 'a'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('c', 'c')]),
            ('out of order',
             [macaroonbakery.Op('c', 'c'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('a', 'a')],
             [macaroonbakery.Op('a', 'a'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('c', 'c')]),
            ('with duplicates',
             [macaroonbakery.Op('c', 'c'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('a', 'a'), macaroonbakery.Op('c', 'a'),
              macaroonbakery.Op('c', 'b'), macaroonbakery.Op('c', 'c'),
              macaroonbakery.Op('a', 'a')],
             [macaroonbakery.Op('a', 'a'), macaroonbakery.Op('a', 'b'),
              macaroonbakery.Op('c', 'a'), macaroonbakery.Op('c', 'b'),
              macaroonbakery.Op('c', 'c')]),
            ('make sure we\'ve got the fields right',
             [macaroonbakery.Op(entity='read', action='two'),
              macaroonbakery.Op(entity='read', action='one'),
              macaroonbakery.Op(entity='write', action='one')],
             [macaroonbakery.Op(entity='read', action='one'),
              macaroonbakery.Op(entity='read', action='two'),
              macaroonbakery.Op(entity='write', action='one')])
        )
        for about, ops, expected in canonical_ops_tests:
            new_ops = copy.copy(ops)
            canonical_ops = macaroonbakery.canonical_ops(new_ops)
            self.assertEquals(canonical_ops, expected)
            # Verify that the original array isn't changed.
            self.assertEquals(new_ops, ops)

    def test_multiple_ops(self):
        test_oven = macaroonbakery.Oven(
            ops_store=macaroonbakery.MemoryOpsStore())
        ops = [macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('one', 'write'),
               macaroonbakery.Op('two', 'read')]
        m = test_oven.macaroon(macaroonbakery.LATEST_BAKERY_VERSION, AGES,
                               None, ops)
        got_ops, conds = test_oven.macaroon_ops([m.macaroon])
        self.assertEquals(len(conds), 1)  # time-before caveat.
        self.assertEquals(macaroonbakery.canonical_ops(got_ops), ops)

    def test_multiple_ops_in_id(self):
        test_oven = macaroonbakery.Oven()
        ops = [macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('one', 'write'),
               macaroonbakery.Op('two', 'read')]
        m = test_oven.macaroon(macaroonbakery.LATEST_BAKERY_VERSION, AGES,
                               None, ops)
        got_ops, conds = test_oven.macaroon_ops([m.macaroon])
        self.assertEquals(len(conds), 1)  # time-before caveat.
        self.assertEquals(macaroonbakery.canonical_ops(got_ops), ops)

    def test_multiple_ops_in_id_with_version1(self):
        test_oven = macaroonbakery.Oven()
        ops = [macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('one', 'write'),
               macaroonbakery.Op('two', 'read')]
        m = test_oven.macaroon(macaroonbakery.BAKERY_V1, AGES, None, ops)
        got_ops, conds = test_oven.macaroon_ops([m.macaroon])
        self.assertEquals(len(conds), 1)  # time-before caveat.
        self.assertEquals(macaroonbakery.canonical_ops(got_ops), ops)

    def test_huge_number_of_ops_gives_small_macaroon(self):
        test_oven = macaroonbakery.Oven(
            ops_store=macaroonbakery.MemoryOpsStore())
        ops = []
        for i in range(30000):
            ops.append(macaroonbakery.Op(entity='entity{}'.format(i),
                                         action='action{}'.format(i)))

        m = test_oven.macaroon(macaroonbakery.LATEST_BAKERY_VERSION, AGES,
                               None, ops)
        got_ops, conds = test_oven.macaroon_ops([m.macaroon])
        self.assertEquals(len(conds), 1)  # time-before caveat.
        self.assertEquals(macaroonbakery.canonical_ops(got_ops),
                          macaroonbakery.canonical_ops(ops))

        data = m.serialize_json()
        self.assertLess(len(data), 300)

    def test_ops_stored_only_once(self):
        st = macaroonbakery.MemoryOpsStore()
        test_oven = macaroonbakery.Oven(ops_store=st)

        ops = [macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('one', 'write'),
               macaroonbakery.Op('two', 'read')]

        m = test_oven.macaroon(macaroonbakery.LATEST_BAKERY_VERSION, AGES,
                               None, ops)
        got_ops, conds = test_oven.macaroon_ops([m.macaroon])
        self.assertEquals(macaroonbakery.canonical_ops(got_ops),
                          macaroonbakery.canonical_ops(ops))

        # Make another macaroon containing the same ops in a different order.
        ops = [macaroonbakery.Op('one', 'write'),
               macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('one', 'read'),
               macaroonbakery.Op('two', 'read')]
        test_oven.macaroon(macaroonbakery.LATEST_BAKERY_VERSION, AGES, None,
                           ops)
        self.assertEquals(len(st._store), 1)