1#! /usr/bin/env python
2
3"""
4SVN helper script.
5
6Try to set the svn:eol-style property to "native" on every .py, .txt, .c and
7.h file in the directory tree rooted at the current directory.
8
9Files with the svn:eol-style property already set (to anything) are skipped.
10
11svn will itself refuse to set this property on a file that's not under SVN
12control, or that has a binary mime-type property set.  This script inherits
13that behavior, and passes on whatever warning message the failing "svn
14propset" command produces.
15
16In the Python project, it's safe to invoke this script from the root of
17a checkout.
18
19No output is produced for files that are ignored.  For a file that gets
20svn:eol-style set, output looks like:
21
22    property 'svn:eol-style' set on 'Lib\ctypes\__init__.py'
23
24For a file not under version control:
25
26    svn: warning: 'patch-finalizer.txt' is not under version control
27
28and for a file with a binary mime-type property:
29
30    svn: File 'Lib\test\test_pep263.py' has binary mime type property
31"""
32
33import re
34import os
35
36def propfiles(root, fn):
37    default = os.path.join(root, ".svn", "props", fn+".svn-work")
38    try:
39        format = int(open(os.path.join(root, ".svn", "format")).read().strip())
40    except IOError:
41        return []
42    if format in (8, 9):
43        # In version 8 and 9, committed props are stored in prop-base, local
44        # modifications in props
45        return [os.path.join(root, ".svn", "prop-base", fn+".svn-base"),
46                os.path.join(root, ".svn", "props", fn+".svn-work")]
47    raise ValueError, "Unknown repository format"
48
49def proplist(root, fn):
50    "Return a list of property names for file fn in directory root"
51    result = []
52    for path in propfiles(root, fn):
53        try:
54            f = open(path)
55        except IOError:
56            # no properties file: not under version control,
57            # or no properties set
58            continue
59        while 1:
60            # key-value pairs, of the form
61            # K <length>
62            # <keyname>NL
63            # V length
64            # <value>NL
65            # END
66            line = f.readline()
67            if line.startswith("END"):
68                break
69            assert line.startswith("K ")
70            L = int(line.split()[1])
71            key = f.read(L)
72            result.append(key)
73            f.readline()
74            line = f.readline()
75            assert line.startswith("V ")
76            L = int(line.split()[1])
77            value = f.read(L)
78            f.readline()
79        f.close()
80    return result
81
82possible_text_file = re.compile(r"\.([hc]|py|txt|sln|vcproj)$").search
83
84for root, dirs, files in os.walk('.'):
85    if '.svn' in dirs:
86        dirs.remove('.svn')
87    for fn in files:
88        if possible_text_file(fn):
89            if 'svn:eol-style' not in proplist(root, fn):
90                path = os.path.join(root, fn)
91                os.system('svn propset svn:eol-style native "%s"' % path)
92