Code

Adapt metakit backend to new security scheme.
[roundup.git] / test / test_htmltemplate.py
1 #
2 # Copyright (c) 2001 Richard Jones
3 # This module is free software, and you may redistribute it and/or modify
4 # under the same terms as Python, so long as this copyright message and
5 # disclaimer are retained in their original form.
6 #
7 # This module is distributed in the hope that it will be useful,
8 # but WITHOUT ANY WARRANTY; without even the implied warranty of
9 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
10 #
11 # $Id: test_htmltemplate.py,v 1.19 2002-07-26 08:27:00 richard Exp $ 
13 import unittest, cgi, time, os, shutil
15 from roundup import date, password
16 from roundup.htmltemplate import TemplateFunctions, IndexTemplate, ItemTemplate
17 from roundup.i18n import _
18 from roundup.hyperdb import String, Password, Date, Interval, Link, \
19     Multilink, Boolean, Number
21 class TestClass:
22     def get(self, nodeid, attribute, default=None):
23         if attribute == 'string':
24             return 'Node %s: I am a string'%nodeid
25         elif attribute == 'filename':
26             return 'file.foo'
27         elif attribute == 'date':
28             return date.Date('2000-01-01')
29         elif attribute == 'boolean':
30             return 0
31         elif attribute == 'number':
32             return 1234
33         elif attribute == 'reldate':
34             return date.Date() + date.Interval('- 2y 1m')
35         elif attribute == 'interval':
36             return date.Interval('-3d')
37         elif attribute == 'link':
38             return '1'
39         elif attribute == 'multilink':
40             return ['1', '2']
41         elif attribute == 'password':
42             return password.Password('sekrit')
43         elif attribute == 'key':
44             return 'the key'+nodeid
45         elif attribute == 'html':
46             return '<html>hello, I am HTML</html>'
47         elif attribute == 'multiline':
48             return 'hello\nworld'
49         elif attribute == 'email':
50             return 'test@foo.domain.example'
51     def list(self):
52         return ['1', '2']
53     def filter(self, search_matches, filterspec, sort, group):
54         return ['1', '2']
55     def getprops(self):
56         return {'string': String(), 'date': Date(), 'interval': Interval(),
57             'link': Link('other'), 'multilink': Multilink('other'),
58             'password': Password(), 'html': String(), 'key': String(),
59             'novalue': String(), 'filename': String(), 'multiline': String(),
60             'reldate': Date(), 'email': String(), 'boolean': Boolean(),
61             'number': Number()}
62     def labelprop(self, default_to_id=0):
63         return 'key'
65 class TestDatabase:
66     classes = {'other': TestClass()}
67     def getclass(self, name):
68         return Class()
69     def __getattr(self, name):
70         return Class()
72 class FunctionCase(unittest.TestCase):
73     def setUp(self):
74         ''' Set up the harness for calling the individual tests
75         '''
76         self.tf = tf = TemplateFunctions()
77         tf.nodeid = '1'
78         tf.cl = TestClass()
79         tf.classname = 'test_class'
80         tf.properties = tf.cl.getprops()
81         tf.db = TestDatabase()
83 #    def do_plain(self, property, escape=0):
84     def testPlain_string(self):
85         s = 'Node 1: I am a string'
86         self.assertEqual(self.tf.do_plain('string'), s)
88     def testPlain_password(self):
89         self.assertEqual(self.tf.do_plain('password'), '*encrypted*')
91     def testPlain_html(self):
92         s = '<html>hello, I am HTML</html>'
93         self.assertEqual(self.tf.do_plain('html', escape=0), s)
94         s = cgi.escape(s)
95         self.assertEqual(self.tf.do_plain('html', escape=1), s)
97     def testPlain_date(self):
98         self.assertEqual(self.tf.do_plain('date'), '2000-01-01.00:00:00')
100     def testPlain_interval(self):
101         self.assertEqual(self.tf.do_plain('interval'), '- 3d')
103     def testPlain_link(self):
104         self.assertEqual(self.tf.do_plain('link'), 'the key1')
106     def testPlain_multilink(self):
107         self.assertEqual(self.tf.do_plain('multilink'), 'the key1, the key2')
109     def testPlain_boolean(self):
110         self.assertEqual(self.tf.do_plain('boolean'), 'No')
112     def testPlain_number(self):
113         self.assertEqual(self.tf.do_plain('number'), '1234')
115 #    def do_field(self, property, size=None, showid=0):
116     def testField_string(self):
117         self.assertEqual(self.tf.do_field('string'),
118             '<input name="string" value="Node 1: I am a string" size="30">')
119         self.assertEqual(self.tf.do_field('string', size=10),
120             '<input name="string" value="Node 1: I am a string" size="10">')
122     def testField_password(self):
123         self.assertEqual(self.tf.do_field('password'),
124             '<input type="password" name="password" size="30">')
125         self.assertEqual(self.tf.do_field('password', size=10),
126             '<input type="password" name="password" size="10">')
128     def testField_html(self):
129         self.assertEqual(self.tf.do_field('html'), '<input name="html" '
130             'value="&lt;html&gt;hello, I am HTML&lt;/html&gt;" size="30">')
131         self.assertEqual(self.tf.do_field('html', size=10),
132             '<input name="html" value="&lt;html&gt;hello, I am '
133             'HTML&lt;/html&gt;" size="10">')
135     def testField_date(self):
136         self.assertEqual(self.tf.do_field('date'),
137             '<input name="date" value="2000-01-01.00:00:00" size="30">')
138         self.assertEqual(self.tf.do_field('date', size=10),
139             '<input name="date" value="2000-01-01.00:00:00" size="10">')
141     def testField_interval(self):
142         self.assertEqual(self.tf.do_field('interval'),
143             '<input name="interval" value="- 3d" size="30">')
144         self.assertEqual(self.tf.do_field('interval', size=10),
145             '<input name="interval" value="- 3d" size="10">')
147     def testField_link(self):
148         self.assertEqual(self.tf.do_field('link'), '''<select name="link">
149 <option value="-1">- no selection -</option>
150 <option selected value="1">the key1</option>
151 <option value="2">the key2</option>
152 </select>''')
154     def testField_multilink(self):
155         self.assertEqual(self.tf.do_field('multilink'),
156             '<input name="multilink" size="30" value="the key1,the key2">')
157         self.assertEqual(self.tf.do_field('multilink', size=10),
158             '<input name="multilink" size="10" value="the key1,the key2">')
160     def testField_boolean(self):
161         self.assertEqual(self.tf.do_field('boolean'),
162             '<input type="checkbox" name="boolean" >')
164     def testField_number(self):
165         self.assertEqual(self.tf.do_field('number'),
166             '<input name="number" value="1234" size="30">')
167         self.assertEqual(self.tf.do_field('number', size=10),
168             '<input name="number" value="1234" size="10">')
170 #    def do_multiline(self, property, rows=5, cols=40)
171     def testMultiline_string(self):
172         self.assertEqual(self.tf.do_multiline('multiline'),
173             '<textarea name="multiline" rows="5" cols="40">'
174             'hello\nworld</textarea>')
175         self.assertEqual(self.tf.do_multiline('multiline', rows=10),
176             '<textarea name="multiline" rows="10" cols="40">'
177             'hello\nworld</textarea>')
178         self.assertEqual(self.tf.do_multiline('multiline', cols=10),
179             '<textarea name="multiline" rows="5" cols="10">'
180             'hello\nworld</textarea>')
182     def testMultiline_nonstring(self):
183         s = _('[Multiline: not a string]')
184         self.assertEqual(self.tf.do_multiline('date'), s)
185         self.assertEqual(self.tf.do_multiline('interval'), s)
186         self.assertEqual(self.tf.do_multiline('password'), s)
187         self.assertEqual(self.tf.do_multiline('link'), s)
188         self.assertEqual(self.tf.do_multiline('multilink'), s)
189         self.assertEqual(self.tf.do_multiline('boolean'), s)
190         self.assertEqual(self.tf.do_multiline('number'), s)
192 #    def do_menu(self, property, size=None, height=None, showid=0):
193     def testMenu_nonlinks(self):
194         s = _('[Menu: not a link]')
195         self.assertEqual(self.tf.do_menu('string'), s)
196         self.assertEqual(self.tf.do_menu('date'), s)
197         self.assertEqual(self.tf.do_menu('interval'), s)
198         self.assertEqual(self.tf.do_menu('password'), s)
199         self.assertEqual(self.tf.do_menu('boolean'), s)
200         self.assertEqual(self.tf.do_menu('number'), s)
202     def testMenu_link(self):
203         self.assertEqual(self.tf.do_menu('link'), '''<select name="link">
204 <option value="-1">- no selection -</option>
205 <option selected value="1">the key1</option>
206 <option value="2">the key2</option>
207 </select>''')
208         self.assertEqual(self.tf.do_menu('link', size=6),
209             '''<select name="link">
210 <option value="-1">- no selection -</option>
211 <option selected value="1">the...</option>
212 <option value="2">the...</option>
213 </select>''')
214         self.assertEqual(self.tf.do_menu('link', showid=1),
215             '''<select name="link">
216 <option value="-1">- no selection -</option>
217 <option selected value="1">other1: the key1</option>
218 <option value="2">other2: the key2</option>
219 </select>''')
221     def testMenu_multilink(self):
222         self.assertEqual(self.tf.do_menu('multilink', height=10),
223             '''<select multiple name="multilink" size="10">
224 <option selected value="1">the key1</option>
225 <option selected value="2">the key2</option>
226 </select>''')
227         self.assertEqual(self.tf.do_menu('multilink', size=6, height=10),
228             '''<select multiple name="multilink" size="10">
229 <option selected value="1">the...</option>
230 <option selected value="2">the...</option>
231 </select>''')
232         self.assertEqual(self.tf.do_menu('multilink', showid=1),
233             '''<select multiple name="multilink" size="2">
234 <option selected value="1">other1: the key1</option>
235 <option selected value="2">other2: the key2</option>
236 </select>''')
238 #    def do_link(self, property=None, is_download=0):
239     def testLink_novalue(self):
240         self.assertEqual(self.tf.do_link('novalue'),
241             _('[no %(propname)s]')%{'propname':'novalue'.capitalize()})
243     def testLink_string(self):
244         self.assertEqual(self.tf.do_link('string'),
245             '<a href="test_class1">Node 1: I am a string</a>')
247     def testLink_file(self):
248         self.assertEqual(self.tf.do_link('filename', is_download=1),
249             '<a href="test_class1/file.foo">file.foo</a>')
251     def testLink_date(self):
252         self.assertEqual(self.tf.do_link('date'),
253             '<a href="test_class1">2000-01-01.00:00:00</a>')
255     def testLink_interval(self):
256         self.assertEqual(self.tf.do_link('interval'),
257             '<a href="test_class1">- 3d</a>')
259     def testLink_link(self):
260         self.assertEqual(self.tf.do_link('link'),
261             '<a href="other1">the key1</a>')
263     def testLink_link_id(self):
264         self.assertEqual(self.tf.do_link('link', showid=1),
265             '<a href="other1" title="the key1">1</a>')
267     def testLink_multilink(self):
268         self.assertEqual(self.tf.do_link('multilink'),
269             '<a href="other1">the key1</a>, <a href="other2">the key2</a>')
271     def testLink_multilink_id(self):
272         self.assertEqual(self.tf.do_link('multilink', showid=1),
273             '<a href="other1" title="the key1">1</a>, <a href="other2" title="the key2">2</a>')
275     def testLink_boolean(self):
276         self.assertEqual(self.tf.do_link('boolean'),
277             '<a href="test_class1">No</a>')
279     def testLink_number(self):
280         self.assertEqual(self.tf.do_link('number'),
281             '<a href="test_class1">1234</a>')
283 #    def do_count(self, property, **args):
284     def testCount_nonlinks(self):
285         s = _('[Count: not a Multilink]')
286         self.assertEqual(self.tf.do_count('string'), s)
287         self.assertEqual(self.tf.do_count('date'), s)
288         self.assertEqual(self.tf.do_count('interval'), s)
289         self.assertEqual(self.tf.do_count('password'), s)
290         self.assertEqual(self.tf.do_count('link'), s)
291         self.assertEqual(self.tf.do_count('boolean'), s)
292         self.assertEqual(self.tf.do_count('number'), s)
294     def testCount_multilink(self):
295         self.assertEqual(self.tf.do_count('multilink'), '2')
297 #    def do_reldate(self, property, pretty=0):
298     def testReldate_nondate(self):
299         s = _('[Reldate: not a Date]')
300         self.assertEqual(self.tf.do_reldate('string'), s)
301         self.assertEqual(self.tf.do_reldate('interval'), s)
302         self.assertEqual(self.tf.do_reldate('password'), s)
303         self.assertEqual(self.tf.do_reldate('link'), s)
304         self.assertEqual(self.tf.do_reldate('multilink'), s)
305         self.assertEqual(self.tf.do_reldate('boolean'), s)
306         self.assertEqual(self.tf.do_reldate('number'), s)
308     def testReldate_date(self):
309         self.assertEqual(self.tf.do_reldate('reldate'), '- 2y 1m')
310         interval = date.Interval('- 2y 1m')
311         self.assertEqual(self.tf.do_reldate('reldate', pretty=1),
312             interval.pretty())
314 #    def do_download(self, property):
315     def testDownload_novalue(self):
316         self.assertEqual(self.tf.do_download('novalue'),
317             _('[no %(propname)s]')%{'propname':'novalue'.capitalize()})
319     def testDownload_string(self):
320         self.assertEqual(self.tf.do_download('string'),
321             '<a href="test_class1/Node 1: I am a string">Node 1: '
322             'I am a string</a>')
324     def testDownload_file(self):
325         self.assertEqual(self.tf.do_download('filename', is_download=1),
326             '<a href="test_class1/file.foo">file.foo</a>')
328     def testDownload_date(self):
329         self.assertEqual(self.tf.do_download('date'),
330             '<a href="test_class1/2000-01-01.00:00:00">2000-01-01.00:00:00</a>')
332     def testDownload_interval(self):
333         self.assertEqual(self.tf.do_download('interval'),
334             '<a href="test_class1/- 3d">- 3d</a>')
336     def testDownload_link(self):
337         self.assertEqual(self.tf.do_download('link'),
338             '<a href="other1/the key1">the key1</a>')
340     def testDownload_multilink(self):
341         self.assertEqual(self.tf.do_download('multilink'),
342             '<a href="other1/the key1">the key1</a>, '
343             '<a href="other2/the key2">the key2</a>')
345     def testDownload_boolean(self):
346         self.assertEqual(self.tf.do_download('boolean'),
347             '<a href="test_class1/No">No</a>')
349     def testDownload_number(self):
350         self.assertEqual(self.tf.do_download('number'),
351             '<a href="test_class1/1234">1234</a>')
353 #    def do_checklist(self, property, reverse=0):
354     def testChecklist_nonlinks(self):
355         s = _('[Checklist: not a link]')
356         self.assertEqual(self.tf.do_checklist('string'), s)
357         self.assertEqual(self.tf.do_checklist('date'), s)
358         self.assertEqual(self.tf.do_checklist('interval'), s)
359         self.assertEqual(self.tf.do_checklist('password'), s)
360         self.assertEqual(self.tf.do_checklist('boolean'), s)
361         self.assertEqual(self.tf.do_checklist('number'), s)
363     def testChecklstk_link(self):
364         self.assertEqual(self.tf.do_checklist('link'),
365             '''the key1:<input type="checkbox" checked name="link" value="the key1">
366 the key2:<input type="checkbox"  name="link" value="the key2">
367 [unselected]:<input type="checkbox"  name="link" value="-1">''')
369     def testChecklink_multilink(self):
370         self.assertEqual(self.tf.do_checklist('multilink'),
371             '''the key1:<input type="checkbox" checked name="multilink" value="the key1">
372 the key2:<input type="checkbox" checked name="multilink" value="the key2">''')
374 #    def do_note(self, rows=5, cols=80):
375     def testNote(self):
376         self.assertEqual(self.tf.do_note(), '<textarea name="__note" '
377             'wrap="hard" rows=5 cols=80></textarea>')
379 #    def do_list(self, property, reverse=0):
380     def testList_nonlinks(self):
381         s = _('[List: not a Multilink]')
382         self.assertEqual(self.tf.do_list('string'), s)
383         self.assertEqual(self.tf.do_list('date'), s)
384         self.assertEqual(self.tf.do_list('interval'), s)
385         self.assertEqual(self.tf.do_list('password'), s)
386         self.assertEqual(self.tf.do_list('link'), s)
387         self.assertEqual(self.tf.do_list('boolean'), s)
388         self.assertEqual(self.tf.do_list('number'), s)
390     def testList_multilink(self):
391         # TODO: test this (needs to have lots and lots of support!
392         #self.assertEqual(self.tf.do_list('multilink'),'')
393         pass
395     def testClasshelp(self):
396         self.assertEqual(self.tf.do_classhelp('theclass', 'prop1,prop2'),
397             '<a href="javascript:help_window(\'classhelp?classname=theclass'
398             '&properties=prop1,prop2\', \'400\', \'400\')"><b>(?)</b></a>')
400 #    def do_email(self, property, rows=5, cols=40)
401     def testEmail_string(self):
402         self.assertEqual(self.tf.do_email('email'), 'test at foo domain example')
404     def testEmail_nonstring(self):
405         s = _('[Email: not a string]')
406         self.assertEqual(self.tf.do_email('date'), s)
407         self.assertEqual(self.tf.do_email('interval'), s)
408         self.assertEqual(self.tf.do_email('password'), s)
409         self.assertEqual(self.tf.do_email('link'), s)
410         self.assertEqual(self.tf.do_email('multilink'), s)
411         self.assertEqual(self.tf.do_email('boolean'), s)
412         self.assertEqual(self.tf.do_email('number'), s)
415 from test_db import setupSchema, MyTestCase, config
417 class Client:
418     user = 'admin'
420 class IndexTemplateCase(unittest.TestCase):
421     def setUp(self):
422         from roundup.backends import anydbm
423         # remove previous test, ignore errors
424         if os.path.exists(config.DATABASE):
425             shutil.rmtree(config.DATABASE)
426         os.makedirs(config.DATABASE + '/files')
427         self.db = anydbm.Database(config, 'test')
428         setupSchema(self.db, 1, anydbm)
430         client = Client()
431         client.db = self.db
432         client.instance = None
433         self.tf = tf = IndexTemplate(client, '', 'issue')
434         tf.props = ['title']
436         # admin user
437         self.db.user.create(username="admin", roles='Admin')
438         self.db.user.create(username="anonymous", roles='User')
440     def testBasic(self):
441         self.assertEqual(self.tf.execute_template('hello'), 'hello')
443     def testValue(self):
444         self.tf.nodeid = self.db.issue.create(title="spam", status='1')
445         self.assertEqual(self.tf.execute_template('<display call="plain(\'title\')">'), 'spam')
447     def testColumnSelection(self):
448         self.tf.nodeid = self.db.issue.create(title="spam", status='1')
449         self.assertEqual(self.tf.execute_template('<property name="title">'
450             '<display call="plain(\'title\')"></property>'
451             '<property name="bar">hello</property>'), 'spam')
452         self.tf.props = ['bar']
453         self.assertEqual(self.tf.execute_template('<property name="title">'
454             '<display call="plain(\'title\')"></property>'
455             '<property name="bar">hello</property>'), 'hello')
457     def testSecurityPass(self):
458         self.assertEqual(self.tf.execute_template(
459             '<require permission="Edit">hello<else>foo</require>'), 'hello')
461     def testSecurityPassValue(self):
462         self.tf.nodeid = self.db.issue.create(title="spam", status='1')
463         self.assertEqual(self.tf.execute_template(
464             '<require permission="Edit">'
465             '<display call="plain(\'title\')">'
466             '<else>not allowed</require>'), 'spam')
468     def testSecurityFail(self):
469         self.tf.client.user = 'anonymous'
470         self.assertEqual(self.tf.execute_template(
471             '<require permission="Edit">hello<else>foo</require>'), 'foo')
473     def testSecurityFailValue(self):
474         self.tf.nodeid = self.db.issue.create(title="spam", status='1')
475         self.tf.client.user = 'anonymous'
476         self.assertEqual(self.tf.execute_template(
477             '<require permission="Edit">allowed<else>'
478             '<display call="plain(\'title\')"></require>'), 'spam')
480     def tearDown(self):
481         if os.path.exists('_test_dir'):
482             shutil.rmtree('_test_dir')
485 class ItemTemplateCase(unittest.TestCase):
486     def setUp(self):
487         ''' Set up the harness for calling the individual tests
488         '''
489         from roundup.backends import anydbm
490         # remove previous test, ignore errors
491         if os.path.exists(config.DATABASE):
492             shutil.rmtree(config.DATABASE)
493         os.makedirs(config.DATABASE + '/files')
494         self.db = anydbm.Database(config, 'test')
495         setupSchema(self.db, 1, anydbm)
497         client = Client()
498         client.db = self.db
499         client.instance = None
500         self.tf = tf = IndexTemplate(client, '', 'issue')
501         tf.nodeid = self.db.issue.create(title="spam", status='1')
503         # admin user
504         self.db.user.create(username="admin", roles='Admin')
505         self.db.user.create(username="anonymous", roles='User')
507     def testBasic(self):
508         self.assertEqual(self.tf.execute_template('hello'), 'hello')
510     def testValue(self):
511         self.assertEqual(self.tf.execute_template('<display call="plain(\'title\')">'), 'spam')
513     def testSecurityPass(self):
514         self.assertEqual(self.tf.execute_template(
515             '<require permission="Edit">hello<else>foo</require>'), 'hello')
517     def testSecurityPassValue(self):
518         self.assertEqual(self.tf.execute_template(
519             '<require permission="Edit">'
520             '<display call="plain(\'title\')">'
521             '<else>not allowed</require>'), 'spam')
523     def testSecurityFail(self):
524         self.tf.client.user = 'anonymous'
525         self.assertEqual(self.tf.execute_template(
526             '<require permission="Edit">hello<else>foo</require>'), 'foo')
528     def testSecurityFailValue(self):
529         self.tf.client.user = 'anonymous'
530         self.assertEqual(self.tf.execute_template(
531             '<require permission="Edit">allowed<else>'
532             '<display call="plain(\'title\')"></require>'), 'spam')
534     def tearDown(self):
535         if os.path.exists('_test_dir'):
536             shutil.rmtree('_test_dir')
538 def suite():
539     return unittest.TestSuite([
540         unittest.makeSuite(FunctionCase, 'test'),
541         unittest.makeSuite(IndexTemplateCase, 'test'),
542         unittest.makeSuite(ItemTemplateCase, 'test'),
543     ])
547 # $Log: not supported by cvs2svn $
548 # Revision 1.18  2002/07/25 07:14:06  richard
549 # Bugger it. Here's the current shape of the new security implementation.
550 # Still to do:
551 #  . call the security funcs from cgi and mailgw
552 #  . change shipped templates to include correct initialisation and remove
553 #    the old config vars
554 # ... that seems like a lot. The bulk of the work has been done though. Honest :)
556 # Revision 1.17  2002/07/18 23:07:07  richard
557 # Unit tests and a few fixes.
559 # Revision 1.16  2002/07/09 05:20:09  richard
560 #  . added email display function - mangles email addrs so they're not so easily
561 #    scraped from the web
563 # Revision 1.15  2002/07/08 06:39:00  richard
564 # Fixed unit test support class so the tests ran again.
566 # Revision 1.14  2002/05/15 06:37:31  richard
567 # ehem and the unit test
569 # Revision 1.13  2002/04/03 05:54:31  richard
570 # Fixed serialisation problem by moving the serialisation step out of the
571 # hyperdb.Class (get, set) into the hyperdb.Database.
573 # Also fixed htmltemplate after the showid changes I made yesterday.
575 # Unit tests for all of the above written.
577 # Revision 1.12  2002/03/29 19:41:48  rochecompaan
578 #  . Fixed display of mutlilink properties when using the template
579 #    functions, menu and plain.
581 # Revision 1.11  2002/02/21 23:11:45  richard
582 #  . fixed some problems in date calculations (calendar.py doesn't handle over-
583 #    and under-flow). Also, hour/minute/second intervals may now be more than
584 #    99 each.
586 # Revision 1.10  2002/02/21 06:57:39  richard
587 #  . Added popup help for classes using the classhelp html template function.
588 #    - add <display call="classhelp('priority', 'id,name,description')">
589 #      to an item page, and it generates a link to a popup window which displays
590 #      the id, name and description for the priority class. The description
591 #      field won't exist in most installations, but it will be added to the
592 #      default templates.
594 # Revision 1.9  2002/02/15 07:08:45  richard
595 #  . Alternate email addresses are now available for users. See the MIGRATION
596 #    file for info on how to activate the feature.
598 # Revision 1.8  2002/02/06 03:47:16  richard
599 #  . #511586 ] unittest FAIL: testReldate_date
601 # Revision 1.7  2002/01/23 20:09:41  jhermann
602 # Proper fix for failing test
604 # Revision 1.6  2002/01/23 05:47:57  richard
605 # more HTML template cleanup and unit tests
607 # Revision 1.5  2002/01/23 05:10:28  richard
608 # More HTML template cleanup and unit tests.
609 #  - download() now implemented correctly, replacing link(is_download=1) [fixed in the
610 #    templates, but link(is_download=1) will still work for existing templates]
612 # Revision 1.4  2002/01/22 22:46:22  richard
613 # more htmltemplate cleanups and unit tests
615 # Revision 1.3  2002/01/22 06:35:40  richard
616 # more htmltemplate tests and cleanup
618 # Revision 1.2  2002/01/22 00:12:07  richard
619 # Wrote more unit tests for htmltemplate, and while I was at it, I polished
620 # off the implementation of some of the functions so they behave sanely.
622 # Revision 1.1  2002/01/21 11:05:48  richard
623 # New tests for htmltemplate (well, it's a beginning)
627 # vim: set filetype=python ts=4 sw=4 et si