~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 = """\
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
169
.SH "EXAMPLES"
170
See
171
.UR http://bazaar.canonical.com/IntroductionToBzr
172
.BR http://bazaar.canonical.com/IntroductionToBzr
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
173
.SH "ENVIRONMENT"
174
.TP
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
175
.I "BZR_HOME"
176
Per-user \'home\' directory. Default on Unix like systems is
177
.I "~"
178
.TP
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
179
.I "BZRPATH"
180
Path where
181
.B "%(bzrcmd)s"
182
is to look for external command.
183
.TP
184
.I "BZREMAIL"
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
185
E-Mail address of the user. Overrides settings from
186
.I "~/.bazaar/bazaar.conf" and
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
187
.IR "EMAIL" .
188
Example content:
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
189
  John Doe <john@example.com>
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
190
.TP
191
.I "EMAIL"
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
192
E-Mail address of the user. Overridden by the settings in the file
193
.I "~/.bazaar/bazaar.conf"
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
194
and of the environment variable
195
.IR "BZREMAIL" .
196
.SH "FILES"
197
.TP
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
198
.I "~/.bazaar/"
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
199
Directory where all the user\'s settings are stored.
200
.TP
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
201
.I "~/.bazaar/bazaar.conf"
202
Stores default settings like name and email address of the
203
user. Settings in this file override the content of
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
204
.I "EMAIL"
205
environment variable. Example content:
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
206
207
  [DEFAULT]
208
  email=John Doe <john@example.com>
209
  editor=/usr/bin/vim
210
  check_signatures=check-available
211
  create_signatures=when-required
212
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
213
.SH "SEE ALSO"
214
.UR http://www.bazaar-ng.org/
215
.BR http://www.bazaar-ng.org/,
216
.UR http://www.bazaar-ng.org/doc/
1185.73.1 by Hans Ulrich Niedermann
Adapted man page generator and templates to bzr 0.6:
217
.BR http://www.bazaar-ng.org/doc/,
218
.UR http://bazaar.canonical.com/BzrDocumentation
219
.BR http://bazaar.canonical.com/BzrDocumentation
635 by Martin Pool
- manpage generator by Hans Ulrich Niedermann
220
"""
221
222
def main():
223
    t = time.time()
224
    tt = time.gmtime(t)
225
    params = \
226
           { "bzrcmd": "bzr",
227
             "datestamp": time.strftime("%Y-%m-%d",tt),
228
             "timestamp": time.strftime("%Y-%m-%d %H:%M:%S +0000",tt),
229
             "version": bzrlib.__version__,
230
             }
231
232
    clp = CommandListParser(params)
233
    bzrlib.help.help("commands", outfile=HelpReader(clp))
234
    clp.end_parse()
235
236
    filename = "bzr.1"
237
    if len(sys.argv) == 2:
238
        filename = sys.argv[1]
239
    if filename == "-":
240
        outfile = sys.stdout
241
    else:
242
        outfile = open(filename,"w")
243
244
    outfile.write(man_preamble % params)
245
    outfile.write(man_escape(man_head % params))
246
    clp.write_to_manpage(outfile)
247
248
    # FIXME:
249
    #   This doesn't do more than the summary so far.
250
    #outfile.write('.SH "DETAILED COMMAND DESCRIPTION"\n')
251
    #for (command,usage,descr) in clp.command_usage:
252
    #    write_command_details(params, command, usage, descr, outfile = outfile)
253
254
    outfile.write(man_escape(man_foot % params))
255
256
257
if __name__ == '__main__':
258
    main()
259
260
261
#>>> code by HUN