3 # update-lily.py -- lilypond autobuilder
5 # source file of the GNU LilyPond music typesetter
7 # download and rebuild latest lilypond or from specified url
13 * use urllib iso ftplib
15 * more flexible build/ftp/patches/releases paths
18 show only: --command='echo "latest is: %n-%v"'
35 gettext.bindtextdomain ('lilypond', '@localedir@')
36 gettext.textdomain('lilypond')
42 sys.path.append ('@datadir@/python')
44 gettext.bindtextdomain ('lilypond', '@localedir@')
45 gettext.textdomain('lilypond')
49 program_name = 'build-lily'
50 package_name = 'lilypond'
51 help_summary = _("Fetch and rebuild from latest source package")
52 build_root = os.path.join (os.environ ['HOME'], 'usr', 'src')
53 release_dir = build_root + '/releases'
54 patch_dir = build_root + '/patches'
60 [ -d %n-%v ] && exit 1 || true;
65 ln -s %n-%v building &&
67 ./configure --prefix=$HOME/usr && make all web
68 ) >> %n-%v/log.txt 2>&1 &&
74 url = 'file:/home/ftp/pub/gnu/LilyPond/development/lilypond-*.tar.gz'
75 url = 'ftp://appel.lilypond.org/pub/gnu/LilyPond/development/lilypond-*.tar.gz'
76 url = 'ftp://ftp.cs.uu.nl/pub/GNU/LilyPond/development/lilypond-*.tar.gz'
81 # lily_py.py -- options and stuff
83 # source file of the GNU LilyPond music typesetter
85 # BEGIN Library for these?
86 # cut-n-paste from ly2dvi
88 program_version = '@TOPLEVEL_VERSION@'
89 if program_version == '@' + 'TOPLEVEL_VERSION' + '@':
90 program_version = '1.3.142'
93 original_dir = os.getcwd ()
94 temp_dir = '%s.dir' % program_name
99 sys.stdout.write ('%s (GNU LilyPond) %s\n' % (program_name, program_version))
103 sys.stdout.write ('\n')
104 sys.stdout.write (_ ('Copyright (c) %s by' % ' 2001'))
105 sys.stdout.write ('\n')
106 sys.stdout.write (' Han-Wen Nienhuys')
107 sys.stdout.write (' Jan Nieuwenhuizen')
108 sys.stdout.write ('\n')
109 sys.stdout.write (_ (r'''
110 Distributed under terms of the GNU General Public License. It comes with
112 sys.stdout.write ('\n')
115 sys.stderr.write (s + '\n')
118 sys.stderr.write (_ ("warning: ") + s)
119 sys.stderr.write ('\n')
123 sys.stderr.write (_ ("error: ") + s)
124 sys.stderr.write ('\n')
125 raise _ ("Exiting ... ")
127 def getopt_args (opts):
128 '''Construct arguments (LONG, SHORT) for getopt from list of options.'''
143 def option_help_str (o):
144 '''Transform one option description (4-tuple ) into neatly formatted string'''
162 return ' ' + sh + sep + long + arg
165 def options_help_str (opts):
166 '''Convert a list of options into a neatly formatted string'''
172 s = option_help_str (o)
173 strs.append ((s, o[3]))
179 str = str + '%s%s%s\n' % (s[0], ' ' * (w - len(s[0]) + 3), s[1])
183 sys.stdout.write (_ ("Usage: %s [OPTION]... FILE") % program_name)
184 sys.stdout.write ('\n\n')
185 sys.stdout.write (help_summary)
186 sys.stdout.write ('\n\n')
187 sys.stdout.write (_ ("Options:"))
188 sys.stdout.write ('\n')
189 sys.stdout.write (options_help_str (option_definitions))
190 sys.stdout.write ('\n\n')
191 sys.stdout.write (_ ("Report bugs to %s") % 'bug-gnu-music@gnu.org')
192 sys.stdout.write ('\n')
198 if not keep_temp_dir_p:
199 temp_dir = tempfile.mktemp (program_name)
201 os.mkdir (temp_dir, 0777)
206 def system (cmd, ignore_error = 0):
208 progress (_ ("Invoking `%s\'") % cmd)
211 msg = ( _ ("error: ") + _ ("command exited with value %d") % st)
213 sys.stderr.write (msg + ' ' + _ ("(ignored)") + ' ')
221 if not keep_temp_dir_p:
223 progress (_ ("Cleaning `%s'...") % temp_dir)
224 system ('rm -rf %s' % temp_dir)
227 def set_setting (dict, key, val):
229 val = string.atof (val)
231 #warning (_ ("invalid value: %s") % `val`)
235 dict[key].append (val)
237 warning (_ ("no such setting: %s") % `key`)
242 option_definitions = [
243 ('DIR', 'b', 'build-root', _ ("unpack and build in DIR [%s]") % build_root),
244 ('COMMAND', 'c', 'command', _ ("execute COMMAND, subtitute:") \
245 + '\n ' + _ ("%b: build root") \
246 + '\n ' + _ ("%n: package name") \
247 + '\n ' + _ ("%r: release directory") \
248 + '\n ' + _ ("%t: tarball") \
249 + '\n ' + _ ("%v: package version") \
251 ('', 'h', 'help', _ ("this help")),
252 ('', 'k', 'keep', _ ("keep all output, and name the directory %s") % temp_dir),
253 ('EMAIL', 'n', 'notify', _ ("upon failure notify EMAIL[,EMAIL]")),
254 ('', 'r', 'remove-previous', _ ("remove previous build")),
255 ('', 'V', 'verbose', _ ("verbose")),
256 ('', 'v', 'version', _ ("print version number")),
257 ('URL', 'u', 'url', _ ("fetch and build URL [%s]") % url),
258 ('', 'w', 'warranty', _ ("show warranty and copyright")),
261 def list_file (user, passwd, host, dir, file):
263 for i in os.listdir (dir):
264 if fnmatch.fnmatch (i, file):
270 def list_ftp (user, passwd, host, dir, file):
274 passwd = program_name
276 ftp = ftplib.FTP (host)
277 ftp.login (user, passwd)
280 list = ftp.nlst (file)
288 m = re.match ('([^:/]*)(:)?(/*([^:]*):)?(/*([^@]*)@)?(//([^/]*))?(.*)/(.*)',
291 error ("can't parse url: %s " % url)
292 return (m.group (1), m.group (4), m.group (6), m.group (8),
293 m.group (9), m.group (10))
296 s = "list_%s ('%s', '%s', '%s', '%s', '%s')" % split_url (url)
299 def version_tuple_to_str (t):
301 my = '.%s%d' % (t[3], t[4])
304 return ('%d.%d.%d' % t[0:3]) + my
306 def version_str_to_tuple (s):
307 t = string.split (s, '.')
310 my_number = string.atoi (t[3][-1])
314 return (string.atoi (t[0]), string.atoi (t[1]), string.atoi (t[2]),
317 def split_package (p):
318 m = re.match ('(.*)-([0-9]*.*).tar.gz', p)
319 return (m.group (1), version_str_to_tuple (m.group (2)))
321 def join_package (t):
322 return t[0] + '-' + version_tuple_to_str (t[1])
324 def copy_file (user, passwd, host, dir, file):
325 os.system ('cp %s/%s .' % (dir, file))
329 def copy_ftp (user, passwd, host, dir, file):
333 passwd = program_name
335 ftp = ftplib.FTP (host)
336 ftp.login (user, passwd)
338 t = tempfile.mktemp (program_name)
341 ftp.retrbinary ('RETR %s/%s' % (dir, file),
342 lambda x, f=f: f.write (x))
344 # huh? Invalid cross-device link
345 # os.rename (t, file)
346 system ('mv %s %s' % (t, file))
358 def copy_url (url, dir):
360 s = "copy_%s ('%s', '%s', '%s', '%s', '%s')" % split_url (url)
364 def find_latest (url):
365 progress (_ ("Listing `%s'...") % url)
366 list = map (split_package, list_url (url))
368 return join_package (list[-1])
371 tar_ball = p + '.tar.gz'
372 (tar_name, tar_version) = split_package (tar_ball)
378 '%v' : version_tuple_to_str (tar_version),
383 for i in expand.keys ():
384 c = re.sub (i, expand[i], c)
387 (sh, long) = getopt_args (__main__.option_definitions)
389 (options, files) = getopt.getopt (sys.argv[1:], sh, long)
400 elif o == '--help' or o == '-h':
402 elif o == '--buid-root' or o == '-b':
404 elif o == '--command' or o == '-c':
406 elif o == '--notify' or o == '-n':
408 elif o == '--remove-previous' or o == '-r':
409 remove_previous_p = 1
410 elif o == '--url' or o == '-u':
412 elif o == '--verbose' or o == '-V':
414 elif o == '--version' or o == '-v':
417 elif o == '--warranty' or o == '-w':
422 latest = find_latest (url)
424 # if os.path.isdir ('%s/%s' % (build_root, latest)):
425 if os.path.exists ('%s/%s/index.html' % (build_root, latest)):
426 progress (_ ("latest is: %s") % latest)
427 progress (_ ("relax, %s is up to date" % package_name))
430 get_base = url[:string.rindex (url, '/')] + '/'
431 if os.path.isdir (patch_dir):
433 if not os.path.isfile (latest + '.diff.gz'):
434 get = get_base + latest + '.diff.gz'
435 progress (_ ("Fetching `%s'...") % get)
438 if not os.path.isdir (build_root):
439 build_root = temp_dir
441 if not os.path.isdir (release_dir):
442 release_dir = temp_dir
445 os.chdir (release_dir)
446 if not os.path.isfile (latest + '.tar.gz'):
447 get = get_base + latest + '.tar.gz'
448 progress (_ ("Fetching `%s'...") % get)
451 if os.path.isdir (os.path.join (build_root, package_name)):
452 os.chdir (os.path.join (build_root, package_name))
453 previous = os.getcwd ()
457 progress (_ ("Building `%s'...") % latest)
458 os.chdir (build_root)
459 if not build (latest):
460 if previous and remove_previous_p:
461 system ('rm -rf %s' % os.path.join (build_root, previous))
464 system ('(date; uname -a) | mail -s "%s failed" %s' % (program_name, notify))
467 os.chdir (original_dir)
468 if release_dir != temp_dir: