• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/python2
2# Copyright (c) 2011 The Chromium OS Authors. All rights reserved.
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
6import dbus, logging
7from autotest_lib.client.common_lib import error
8
9sample = {
10    'pdu' :
11      '07914140540510F0040B916171056429F500001190804181106904D4F29C0E',
12    'parsed' :
13      {'text' : 'Test',
14       'number' : '+16175046925',
15       'timestamp' : '110908141801-04',
16       'smsc' : '+14044550010'
17       }
18    }
19
20sample_multipart = {
21    'pdu' :
22      ['07912160130320F8440B916171056429F5000011909161037469A0050003920201A9'
23       'E5391DF43683E6EF7619C47EBBCF207A194F0789EB74D03D4D47BFEB7450D89D0791'
24       'D366737A5C67D3416374581E1ED3CBF23928ED1EB3EBE43219947683E8E832A85D9E'
25       'CFC3E7B20B4445A7E72077B94C9E83E86F90B80C7ADBCB72101D5D06B1CBEE331D0D'
26       'A2A3E5E539FACD2683CC6F39888E2E83D8EF71980D9ABFCDF47B585E06D1DF',
27       '07912160130320F5440B916171056429F50000119091610384691505000392020241'
28       'E437888E2E83E670769AEE02'],
29    'parsed' :
30      {'text' : 'Test of some long text but without any difficult characters'
31       ' included in the message. This needs to be over the length threshold'
32       ' for the local software to do the split.',
33       'number' : '+16175046925',
34       'timestamp' : '110919163047-04',
35       'smsc' : '+12063130028'
36       }
37    }
38
39
40class SmsStore(object):
41    '''SMS content management - this maintains an internal model of the
42    index->PDU mapping that the fakemodem program should be returning so
43    that tests can add and remove individual PDUs and handles generating
44    the correct set of responses, including the complete SMS list.
45    '''
46
47    def __init__(self, fakemodem):
48        self.fakemodem = fakemodem
49        self.smsdict = {}
50        self.fakemodem.SetResponse('\+CMGR=', '', '+CMS ERROR: 321')
51        self.fakemodem.SetResponse('\+CMGD=', '', '+CMS ERROR: 321')
52        self._sms_regen_list()
53
54    def sms_insert(self, index, pdu):
55        '''Add a SMS to the fake modem's list.'''
56        smsc_len = int(pdu[0:1], 16)
57        mlen = len(pdu)/2 - smsc_len - 1
58
59        self.fakemodem.RemoveResponse('\+CMGD=')
60        self.fakemodem.RemoveResponse('\+CMGR=')
61        self.fakemodem.SetResponse('\+CMGD=%d' % (index), '', '')
62        self.fakemodem.SetResponse('\+CMGR=%d' % (index),
63                                   '+CMGR: 1,,%d\r\n%s' % (mlen, pdu), '')
64        self.fakemodem.SetResponse('\+CMGR=', '', '+CMS ERROR: 321')
65        self.fakemodem.SetResponse('\+CMGD=', '', '+CMS ERROR: 321')
66
67        self.smsdict[index] = pdu
68        self._sms_regen_list()
69
70    def sms_receive(self, index, pdu):
71        '''Add a SMS to the fake modem's list, like sms_insert(), and generate
72        an unsolicited new-sms message.'''
73        self.sms_insert(index, pdu)
74        self.fakemodem.SendUnsolicited('+CMTI: "ME",%d'%(index))
75
76    def sms_remove(self, index):
77        '''Remove a SMS from the fake modem's list'''
78        self.fakemodem.RemoveResponse('\+CMGR=%d' % (index))
79        self.fakemodem.RemoveResponse('\+CMGD=%d' % (index))
80        del self.smsdict[index]
81        self._sms_regen_list()
82
83    def _sms_regen_list(self):
84        response = ''
85        keys = self.smsdict.keys()
86        keys.sort()
87        for i in keys:
88            pdu = self.smsdict[i]
89            smsc_len = int(pdu[0:1],16)
90            mlen = len(pdu)/2 - smsc_len - 1
91            response = response + '+CMGL: %d,1,,%d\r\n%s\r\n' % (i, mlen, pdu)
92        self.fakemodem.SetResponse('\+CMGL=4', response, '')
93
94
95class SmsTest(object):
96    def __init__(self, gsmsms):
97        self.gsmsms = gsmsms
98
99    def compare(self, expected, got):
100        '''Compare two SMS dictionaries, discounting the index number if
101        not specified in the first.'''
102        if expected == got:
103            return True
104        if 'index' in expected:
105            return False
106        if 'index' not in got:
107            return False
108        got = dict(got)
109        del got['index']
110        return expected == got
111
112    def compare_list(self, expected_list, got_list):
113        if len(expected_list) != len(got_list):
114            return False
115        # There must be a more Pythonic way to do this
116        for (expected,got) in zip(expected_list, got_list):
117            if self.compare(expected, got) == False:
118                return False
119        return True
120
121    def test_get(self, index, expected):
122        try:
123            sms = self.gsmsms.Get(index)
124        except dbus.DBusException, db:
125            if expected is not None:
126                raise
127            return
128
129        if expected is None:
130            logging.info('Got %s' % sms)
131            raise error.TestFail('SMS.Get(%d) succeeded unexpectedly' %
132                                 index)
133        if self.compare(expected, sms) == False:
134            logging.info('Got %s, expected %s' % (sms, expected))
135            raise error.TestFail('SMS.Get(%d) did not match expected values' %
136                                 index)
137
138    def test_delete(self, index, expected_success):
139        try:
140            self.gsmsms.Delete(index)
141            if expected_success == False:
142                raise error.TestFail('SMS.Delete(%d) succeeded unexpectedly' %
143                                     index)
144        except dbus.DBusException, db:
145            if expected_success:
146                raise
147
148    def test_list(self, expected_list):
149        sms_list = self.gsmsms.List()
150        if self.compare_list(expected_list, sms_list) == False:
151            logging.info('Got %s, expected %s' % (sms_list, expected_list))
152            raise error.TestFail('SMS.List() did not match expected values')
153
154    def test_has_none(self):
155        '''Test that the SMS interface has no messages.'''
156        self.test_list([])
157        self.test_get(1, None)
158        self.test_delete(1, False)
159        self.test_delete(2, False)
160
161    def test_has_one(self, parsed_sms):
162        '''Test that the SMS interface has exactly one message at index 1
163        As a side effect, deletes the message.'''
164        self.test_list([parsed_sms])
165        self.test_get(1, parsed_sms)
166        self.test_get(2, None)
167        self.test_delete(2, False)
168        self.test_delete(1, True)
169