summaryrefslogtreecommitdiff
path: root/examples/statemachine/libraryBookDemo.py
blob: 84108e56a67a30110f43447875af01ce3fd5b825 (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
import statemachine
import librarybookstate


class Book:
    def __init__(self):
        self._state = librarybookstate.New()

    @property
    def state(self):
        return self._state

    # get behavior/properties from current state
    def __getattr__(self, attrname):
        attr = getattr(self._state, attrname)
        if isinstance(getattr(librarybookstate, attrname, None),
                      librarybookstate.BookStateTransition):
            return lambda: setattr(self, '_state', attr())
        return attr

    def __str__(self):
        return "{}: {}".format(self.__class__.__name__, self._state)


class RestrictedBook(Book):
    def __init__(self):
        super(RestrictedBook, self).__init__()
        self._authorized_users = []

    def authorize(self, name):
        self._authorized_users.append(name)

    # specialized checkout to check permission of user first
    def checkout(self, user=None):
        if user in self._authorized_users:
            self._state = self._state.checkout()
        else:
            raise Exception("{} could not check out restricted book".format((user, "anonymous")[user is None]))


def run_demo():
    book = Book()
    book.shelve()
    print(book)
    book.checkout()
    print(book)
    book.checkin()
    print(book)
    book.reserve()
    print(book)
    try:
        book.checkout()
    except statemachine.InvalidTransitionException:
        print('..cannot check out reserved book')
    book.release()
    print(book)
    book.checkout()
    print(book)
    print()

    restricted_book = RestrictedBook()
    restricted_book.authorize("BOB")
    restricted_book.restrict()
    print(restricted_book)
    for name in [None, "BILL", "BOB"]:
        try:
            restricted_book.checkout(name)
        except Exception as e:
            print('..' + str(e))
        else:
            print('checkout to', name)
    print(restricted_book)
    restricted_book.checkin()
    print(restricted_book)


if __name__ == '__main__':
    run_demo()