Statistics
| Branch: | Tag: | Revision:

root / kamaki / cli / history.py @ 5e383dd4

History | View | Annotate | Download (3.8 kB)

1
#!/usr/bin/env python
2

    
3
# Copyright 2012-2014 GRNET S.A. All rights reserved.
4
#
5
# Redistribution and use in source and binary forms, with or
6
# without modification, are permitted provided that the following
7
# conditions are met:
8
#
9
#   1. Redistributions of source code must retain the above
10
#      copyright notice, this list of conditions and the following
11
#      disclaimer.
12
#
13
#   2. Redistributions in binary form must reproduce the above
14
#      copyright notice, this list of conditions and the following
15
#      disclaimer in the documentation and/or other materials
16
#      provided with the distribution.
17
#
18
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
19
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
22
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
25
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
26
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
28
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29
# POSSIBILITY OF SUCH DAMAGE.
30
#
31
# The views and conclusions contained in the software and
32
# documentation are those of the authors and should not be
33
# interpreted as representing official policies, either expressed
34
# or implied, of GRNET S.A.
35

    
36
import codecs
37

    
38

    
39
class History(object):
40
    def __init__(self, filepath, token=None):
41
        self.filepath = filepath
42
        self.token = token
43
        self._limit = 0
44
        self.counter = 0
45

    
46
    def __getitem__(self, cmd_ids):
47
        with codecs.open(self.filepath, mode='r', encoding='utf-8') as f:
48
            try:
49
                cmd_list = f.readlines()[1:]
50
                return cmd_list[cmd_ids]
51
            except IndexError:
52
                return None
53

    
54
    @property
55
    def limit(self):
56
        return self._limit
57

    
58
    @limit.setter
59
    def limit(self, new_limit):
60
        new_limit = int(new_limit)
61
        if new_limit < 0:
62
            raise ValueError('Invalid history limit (%s)' % new_limit)
63
        old_limit, self._limit = self._limit, new_limit
64
        if self._limit and self._limit < old_limit:
65
            with codecs.open(self.filepath, mode='r', encoding='utf-8') as f:
66
                lines = f.readlines()
67
                old_len = len(lines[1:])
68
                if old_len > new_limit:
69
                    self.counter += old_len - new_limit
70
                    f.write('%s\n' % self.counter)
71
                    f.write(''.join(lines[(self.counter + 1):]))
72
                    f.flush()
73

    
74
    @classmethod
75
    def _match(self, line, match_terms):
76
        if match_terms:
77
            return all(term in line for term in match_terms.split())
78
        return True
79

    
80
    def get(self, match_terms=None, limit=0):
81
        """DEPRECATED since 0.14"""
82
        limit = int(limit or 0)
83
        r = ['%s.\t%s' % (i + 1, line) for i, line in enumerate(self[:]) if (
84
                self._match(line, match_terms))]
85
        return r[- limit:]
86

    
87
    def add(self, line):
88
        line = line.replace(self.token, '...') if self.token else line
89
        with codecs.open(self.filepath, mode='a+', encoding='utf-8') as f:
90
            f.write(line + '\n')
91
            f.flush()
92
        self.limit = self.limit
93

    
94
    def empty(self):
95
        with open(self.filepath, 'w') as f:
96
            f.write('0\n')
97
            f.flush()
98
        self.counter = 0
99

    
100
    def clean(self):
101
        """DEPRECATED since version 0.14"""
102
        return self.empty()
103

    
104
    def retrieve(self, cmd_id):
105
        if not cmd_id:
106
            return None
107
        cmd_id = int(cmd_id)
108
        return self[cmd_id - (1 if cmd_id > 0 else 0)]