VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/db/gen-sql-comments.py@ 57044

Last change on this file since 57044 was 56295, checked in by vboxsync, 10 years ago

ValidationKit: Updated (C) year.

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 7.8 KB
Line 
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3# $Id: gen-sql-comments.py 56295 2015-06-09 14:29:55Z vboxsync $
4
5"""
6Copyright (C) 2012-2015 Oracle Corporation
7
8This file is part of VirtualBox Open Source Edition (OSE), as
9available from http://www.virtualbox.org. This file is free software;
10you can redistribute it and/or modify it under the terms of the GNU
11General Public License (GPL) as published by the Free Software
12Foundation, in version 2 as it comes in the "COPYING" file of the
13VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15
16The contents of this file may alternatively be used under the terms
17of the Common Development and Distribution License Version 1.0
18(CDDL) only, as it comes in the "COPYING.CDDL" file of the
19VirtualBox OSE distribution, in which case the provisions of the
20CDDL are applicable instead of those of the GPL.
21
22You may elect to license modified versions of this file under the
23terms and conditions of either the GPL or the CDDL or both.
24"""
25"""
26Converts doxygen style comments in SQL script to COMMENT ON statements.
27"""
28
29import sys;
30import re;
31
32
33def errorMsg(sMsg):
34 sys.stderr.write('error: %s\n' % (sMsg,));
35 return 1;
36
37class SqlDox(object):
38 """
39 Class for parsing relevant comments out of a pgsql file
40 and emit COMMENT ON statements from it.
41 """
42
43 def __init__(self, oFile, sFilename):
44 self.oFile = oFile;
45 self.sFilename = sFilename;
46 self.iLine = 0; # The current input line number.
47 self.sComment = None; # The current comment.
48 self.fCommentComplete = False; # Indicates that the comment has ended.
49 self.sCommentSqlObj = None; # SQL object indicated by the comment (@table).
50 self.sOuterSqlObj = None; # Like 'table yyyy' or 'type zzzz'.
51 self.sPrevSqlObj = None; # Like 'table xxxx'.
52
53
54 def error(self, sMsg):
55 return errorMsg('%s(%d): %s' % (self.sFilename, self.iLine, sMsg,));
56
57 def dprint(self, sMsg):
58 sys.stderr.write('debug: %s\n' % (sMsg,));
59 return True;
60
61 def resetComment(self):
62 self.sComment = None;
63 self.fCommentComplete = False;
64 self.sCommentSqlObj = None;
65
66 def quoteSqlString(self, s):
67 return s.replace("'", "''");
68
69 def commitComment2(self, sSqlObj):
70 if self.sComment is not None and sSqlObj is not None:
71 print("COMMENT ON %s IS\n '%s';\n\n" % (sSqlObj, self.quoteSqlString(self.sComment.strip())));
72 self.resetComment();
73 return True;
74
75 def commitComment(self):
76 return self.commitComment2(self.sCommentSqlObj);
77
78 def process(self):
79 for sLine in self.oFile:
80 self.iLine += 1;
81
82 sLine = sLine.strip();
83 self.dprint('line %d: %s\n' % (self.iLine, sLine));
84 if sLine.startswith('--'):
85 if sLine.startswith('--- '):
86 #
87 # New comment.
88 # The first list may have a @table, @type or similar that we're interested in.
89 #
90 self.commitComment();
91
92 sLine = sLine.lstrip('- ');
93 if sLine.startswith('@table '):
94 self.sCommentSqlObj = 'TABLE ' + (sLine[7:]).rstrip();
95 self.sComment = '';
96 elif sLine.startswith('@type '):
97 self.sCommentSqlObj = 'TYPE ' + (sLine[6:]).rstrip();
98 self.sComment = '';
99 elif sLine.startswith('@todo') \
100 or sLine.startswith('@file') \
101 or sLine.startswith('@page') \
102 or sLine.startswith('@name') \
103 or sLine.startswith('@{') \
104 or sLine.startswith('@}'):
105 # Ignore.
106 pass;
107 elif sLine.startswith('@'):
108 return self.error('Unknown tag: %s' % (sLine,));
109 else:
110 self.sComment = sLine;
111
112 elif (sLine.startswith('-- ') or sLine == '--') \
113 and self.sComment is not None and self.fCommentComplete is False:
114 #
115 # Append line to comment.
116 #
117 if sLine == '--':
118 sLine = '';
119 else:
120 sLine = (sLine[3:]);
121 if self.sComment == '':
122 self.sComment = sLine;
123 else:
124 self.sComment += "\n" + sLine;
125
126 elif sLine.startswith('--< '):
127 #
128 # Comment that starts on the same line as the object it describes.
129 #
130 sLine = (sLine[4:]).rstrip();
131 # => Later/never.
132 else:
133 #
134 # Not a comment that interests us. So, complete any open
135 # comment and commit it if we know which SQL object it
136 # applies to.
137 #
138 self.fCommentComplete = True;
139 if self.sCommentSqlObj is not None:
140 self.commitComment();
141 else:
142 #
143 # Not a comment. As above, we complete and optionally commit
144 # any open comment.
145 #
146 self.fCommentComplete = True;
147 if self.sCommentSqlObj is not None:
148 self.commitComment();
149
150 #
151 # Check for SQL (very fuzzy and bad).
152 #
153 asWords = sLine.split(' ');
154 if len(asWords) >= 3 \
155 and asWords[0] == 'CREATE':
156 # CREATE statement.
157 sType = asWords[1];
158 sName = asWords[2];
159 if sType == 'UNIQUE' and sName == 'INDEX' and len(asWords) >= 4:
160 sType = asWords[2];
161 sName = asWords[3];
162 if sType in ('TABLE', 'TYPE', 'INDEX', 'VIEW'):
163 self.sOuterSqlObj = sType + ' ' + sName;
164 self.sPrevSqlObj = self.sOuterSqlObj;
165 self.dprint('%s' % (self.sOuterSqlObj,));
166 self.commitComment2(self.sOuterSqlObj);
167 elif len(asWords) >= 1 \
168 and self.sOuterSqlObj is not None \
169 and self.sOuterSqlObj.startswith('TABLE ') \
170 and re.search("^(as|al|bm|c|enm|f|i|l|s|ts|uid|uuid)[A-Z][a-zA-Z0-9]*$", asWords[0]) is not None:
171 # Possibly a column name.
172 self.sPrevSqlObj = 'COLUMN ' + self.sOuterSqlObj[6:] + '.' + asWords[0];
173 self.dprint('column? %s' % (self.sPrevSqlObj));
174 self.commitComment2(self.sPrevSqlObj);
175
176 #
177 # Check for semicolon.
178 #
179 if sLine.find(");") >= 0:
180 self.sOuterSqlObj = None;
181
182 return 0;
183
184
185def usage():
186 sys.stderr.write('usage: gen-sql-comments.py <filename.pgsql>\n'
187 '\n'
188 'The output goes to stdout.\n');
189 return 0;
190
191
192def main(asArgs):
193 # Parse the argument. :-)
194 sInput = None;
195 if (len(asArgs) != 2):
196 sys.stderr.write('syntax error: expected exactly 1 argument, a psql file\n');
197 usage();
198 return 2;
199 sInput = asArgs[1];
200
201 # Do the job, outputting to standard output.
202 try:
203 oFile = open(sInput, 'r');
204 except:
205 return errorMsg("failed to open '%s' for reading" % (sInput,));
206 me = SqlDox(oFile, sInput);
207 return me.process();
208
209sys.exit(main(sys.argv));
210
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette