~bzr-pqm/bzr/bzr.dev

635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
1
#!/usr/bin/python
2
3
# Copyright (C) 2005 by Hans Ulrich Niedermann
4
# Portions Copyright (C) 2005 by Canonical Ltd
5
6
# This program is free software; you can redistribute it and/or modify
7
# it under the terms of the GNU General Public License as published by
8
# the Free Software Foundation; either version 2 of the License, or
9
# (at your option) any later version.
10
11
# This program is distributed in the hope that it will be useful,
12
# but WITHOUT ANY WARRANTY; without even the implied warranty of
13
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
# GNU General Public License for more details.
15
16
# You should have received a copy of the GNU General Public License
17
# along with this program; if not, write to the Free Software
18
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19
20
#<<< code taken from bzr (C) Canonical
21
22
import os, sys
23
24
import bzrlib, bzrlib.help
25
26
#>>> code taken from bzr (C) Canonical
27
28
#<<< code by HUN
29
30
import time
31
import re
32
33
34
def man_escape(string):
35
    result = string.replace("\\","\\\\")
36
    result = result.replace("`","\\`")
37
    result = result.replace("'","\\'")
38
    result = result.replace("-","\\-")
39
    return result
40
41
42
class Parser:
43
44
    def parse_line(self, line):
45
        pass
46
47
48
class CommandListParser(Parser):
49
50
    """Parser for output of "bzr help commands".
51
52
    The parsed content can then be used to
53
    - write a "COMMAND OVERVIEW" section into a man page
54
    - provide a list of all commands
55
    """
56
57
    def __init__(self,params):
58
        self.params = params
59
        self.command_usage = []
60
        self.all_commands = []
61
        self.usage_exp = re.compile("([a-z0-9-]+).*")
62
        self.descr_exp = re.compile("    ([A-Z].*)\s*")
63
        self.state = 0
64
        self.command = None
65
        self.usage = None
66
        self.descr = None
67
68
    def parse_line(self, line):
69
        m = self.usage_exp.match(line)
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
70
	if line == '':
71
		return
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
72
        if m:
73
            if self.state == 0:
74
                if self.usage:
75
                    self.command_usage.append((self.command,self.usage,self.descr))
76
                    self.all_commands.append(self.command)
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
77
                self.usage = " ".join(line.split(" ")[1:])
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
78
                self.command = m.groups()[0]
79
            else:
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
80
                raise RuntimeError, "matching usage line in state %d" % state
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
81
            self.state = 1
82
            return
83
        m = self.descr_exp.match(line)
84
        if m:
85
            if self.state == 1:
86
                self.descr = m.groups()[0]
87
            else:
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
88
                raise RuntimeError, "matching descr line in state %d" % state
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
89
            self.state = 0
90
            return
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
91
        raise RuntimeError, "Cannot parse this line ('%s')." % line
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
92
93
    def end_parse(self):
94
        if self.state == 0:
95
            if self.usage:
96
                self.command_usage.append((self.command,self.usage,self.descr))
97
                self.all_commands.append(self.command)
98
        else:
1185.1.43 by Robert Collins
patch from Rob Weir to correct bzr-man.py
99
            raise RuntimeError, "ending parse in state %d" % state
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
100
101
    def write_to_manpage(self, outfile):
102
        bzrcmd = self.params["bzrcmd"]
103
        outfile.write('.SH "COMMAND OVERVIEW"\n')
104
        for (command,usage,descr) in self.command_usage:
1185.29.1 by Wouter van Heyst
Eliminate double whitespace lines in manpage, reported by liw
105
            outfile.write('.TP\n.B "%s %s"\n%s\n' % (bzrcmd, usage, descr))
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
106
107
108
class HelpReader:
109
110
    def __init__(self, parser):
111
        self.parser = parser
112
113
    def write(self, data):
114
        if data[-1] == '\n':
115
            data = data[:-1]
116
        for line in data.split('\n'):
117
            self.parser.parse_line(line)
118
119
120
def write_command_details(params, command, usage, descr, outfile):
121
    x = ('.SS "%s %s"\n.B "%s"\n.PP\n.B "Usage:"\n%s %s\n\n' %
122
         (params["bzrcmd"],
123
          command,
124
          descr,
125
          params["bzrcmd"],
126
          usage))
127
    outfile.write(man_escape(x))
128
129
130
man_preamble = """\
131
.\\\" Man page for %(bzrcmd)s (bazaar-ng)
132
.\\\"
133
.\\\" Large parts of this file are autogenerated from the output of
134
.\\\"     \"%(bzrcmd)s help commands\"
135
.\\\"     \"%(bzrcmd)s help <cmd>\"
136
.\\\"
137
.\\\" Generation time: %(timestamp)s
138
.\\\"
139
"""
140
141
# The DESCRIPTION text was taken from http://www.bazaar-ng.org/
142
# and is thus (C) Canonical
143
man_head = """\
144
.TH bzr 1 "%(datestamp)s" "%(version)s" "bazaar-ng"
145
.SH "NAME"
146
%(bzrcmd)s - bazaar-ng next-generation distributed version control
147
.SH "SYNOPSIS"
148
.B "%(bzrcmd)s"
149
.I "command"
150
[
151
.I "command_options"
152
]
153
.br
154
.B "%(bzrcmd)s"
155
.B "help"
156
.br
157
.B "%(bzrcmd)s"
158
.B "help"
159
.I "command"
160
.SH "DESCRIPTION"
161
bazaar-ng (or
162
.B "%(bzrcmd)s"
163
) is a project of Canonical to develop an open source distributed version control system that is powerful, friendly, and scalable. Version control means a system that keeps track of previous revisions of software source code or similar information and helps people work on it in teams.
164
.SS "Warning"
165
bazaar-ng is at an early stage of development, and the design is still changing from week to week. This man page here may be inconsistent with itself, with other documentation or with the code, and sometimes refer to features that are planned but not yet written. Comments are still very welcome; please send them to bazaar-ng@lists.canonical.com.
166
"""
167
168
man_foot = """\
169
.SH "ENVIRONMENT"
170
.TP
171
.I "BZRPATH"
172
Path where
173
.B "%(bzrcmd)s"
174
is to look for external command.
175
.TP
176
.I "BZREMAIL"
177
E-Mail address of the user. Overrides
178
.I "~/.bzr.conf/email" and
179
.IR "EMAIL" .
180
Example content:
181
.I "John Doe <john@example.com>"
182
.TP
183
.I "EMAIL"
184
E-Mail address of the user. Overridden by the content of the file
185
.I "~/.bzr.conf/email"
186
and of the environment variable
187
.IR "BZREMAIL" .
188
.SH "FILES"
189
.TP
190
.I "~/.bzr.conf/"
191
Directory where all the user\'s settings are stored.
192
.TP
193
.I "~/.bzr.conf/email"
194
Stores name and email address of the user. Overrides content of
195
.I "EMAIL"
196
environment variable. Example content:
197
.I "John Doe <john@example.com>"
198
.SH "SEE ALSO"
199
.UR http://www.bazaar-ng.org/
200
.BR http://www.bazaar-ng.org/,
201
.UR http://www.bazaar-ng.org/doc/
202
.BR http://www.bazaar-ng.org/doc/
203
"""
204
205
def main():
206
    t = time.time()
207
    tt = time.gmtime(t)
208
    params = \
209
           { "bzrcmd": "bzr",
210
             "datestamp": time.strftime("%Y-%m-%d",tt),
211
             "timestamp": time.strftime("%Y-%m-%d %H:%M:%S +0000",tt),
212
             "version": bzrlib.__version__,
213
             }
214
215
    clp = CommandListParser(params)
216
    bzrlib.help.help("commands", outfile=HelpReader(clp))
217
    clp.end_parse()
218
219
    filename = "bzr.1"
220
    if len(sys.argv) == 2:
221
        filename = sys.argv[1]
222
    if filename == "-":
223
        outfile = sys.stdout
224
    else:
225
        outfile = open(filename,"w")
226
227
    outfile.write(man_preamble % params)
228
    outfile.write(man_escape(man_head % params))
229
    clp.write_to_manpage(outfile)
230
231
    # FIXME:
232
    #   This doesn't do more than the summary so far.
233
    #outfile.write('.SH "DETAILED COMMAND DESCRIPTION"\n')
234
    #for (command,usage,descr) in clp.command_usage:
235
    #    write_command_details(params, command, usage, descr, outfile = outfile)
236
237
    outfile.write(man_escape(man_foot % params))
238
239
240
if __name__ == '__main__':
241
    main()
242
243
244
#>>> code by HUN