netrc.py 5.61 KB
Newer Older
1 2
"""An object-oriented interface to .netrc files."""

Tim Peters's avatar
Tim Peters committed
3
# Module and documentation by Eric S. Raymond, 21 Dec 1998
4

5
import os, shlex, stat
6

7 8 9 10 11 12
__all__ = ["netrc", "NetrcParseError"]


class NetrcParseError(Exception):
    """Exception raised on syntax errors in the .netrc file."""
    def __init__(self, msg, filename=None, lineno=None):
13
        self.filename = filename
14 15 16 17 18 19 20
        self.lineno = lineno
        self.msg = msg
        Exception.__init__(self, msg)

    def __str__(self):
        return "%s (%s, line %s)" % (self.msg, self.filename, self.lineno)

21

22 23
class netrc:
    def __init__(self, file=None):
24
        default_netrc = file is None
25
        if file is None:
26 27 28
            try:
                file = os.path.join(os.environ['HOME'], ".netrc")
            except KeyError:
29
                raise OSError("Could not find .netrc: $HOME is not set")
30 31
        self.hosts = {}
        self.macros = {}
32
        with open(file) as fp:
33
            self._parse(file, fp, default_netrc)
34

35
    def _parse(self, file, fp, default_netrc):
36
        lexer = shlex.shlex(fp)
37
        lexer.wordchars += r"""!"#$%&'()*+,-./:;<=>?@[\]^_`{|}~"""
38
        lexer.commenters = lexer.commenters.replace('#', '')
39 40
        while 1:
            # Look for a machine, default, or macdef top-level keyword
41
            saved_lineno = lexer.lineno
42
            toplevel = tt = lexer.get_token()
43
            if not tt:
44
                break
45
            elif tt[0] == '#':
46 47 48
                if lexer.lineno == saved_lineno and len(tt) == 1:
                    lexer.instream.readline()
                continue
49 50 51 52
            elif tt == 'machine':
                entryname = lexer.get_token()
            elif tt == 'default':
                entryname = 'default'
Tim Peters's avatar
Tim Peters committed
53
            elif tt == 'macdef':                # Just skip to end of macdefs
54 55
                entryname = lexer.get_token()
                self.macros[entryname] = []
56
                lexer.whitespace = ' \t'
57 58
                while 1:
                    line = lexer.instream.readline()
59 60
                    if not line or line == '\012':
                        lexer.whitespace = ' \t\r\n'
61 62
                        break
                    self.macros[entryname].append(line)
63
                continue
64
            else:
65 66
                raise NetrcParseError(
                    "bad toplevel token %r" % tt, file, lexer.lineno)
67 68

            # We're looking at start of an entry for a named machine or default.
69 70
            login = ''
            account = password = None
71
            self.hosts[entryname] = {}
72 73
            while 1:
                tt = lexer.get_token()
74 75
                if (tt.startswith('#') or
                    tt in {'', 'machine', 'default', 'macdef'}):
76
                    if password:
77 78 79 80
                        self.hosts[entryname] = (login, account, password)
                        lexer.push_token(tt)
                        break
                    else:
81 82 83 84
                        raise NetrcParseError(
                            "malformed %s entry %s terminated by %s"
                            % (toplevel, entryname, repr(tt)),
                            file, lexer.lineno)
85 86 87 88 89
                elif tt == 'login' or tt == 'user':
                    login = lexer.get_token()
                elif tt == 'account':
                    account = lexer.get_token()
                elif tt == 'password':
90 91 92
                    if os.name == 'posix' and default_netrc:
                        prop = os.fstat(fp.fileno())
                        if prop.st_uid != os.getuid():
93
                            import pwd
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
                            try:
                                fowner = pwd.getpwuid(prop.st_uid)[0]
                            except KeyError:
                                fowner = 'uid %s' % prop.st_uid
                            try:
                                user = pwd.getpwuid(os.getuid())[0]
                            except KeyError:
                                user = 'uid %s' % os.getuid()
                            raise NetrcParseError(
                                ("~/.netrc file owner (%s) does not match"
                                 " current user (%s)") % (fowner, user),
                                file, lexer.lineno)
                        if (prop.st_mode & (stat.S_IRWXG | stat.S_IRWXO)):
                            raise NetrcParseError(
                               "~/.netrc access too permissive: access"
                               " permissions must restrict access to only"
                               " the owner", file, lexer.lineno)
111 112
                    password = lexer.get_token()
                else:
113 114
                    raise NetrcParseError("bad follower token %r" % tt,
                                          file, lexer.lineno)
115 116

    def authenticators(self, host):
117
        """Return a (user, account, password) tuple for given host."""
118
        if host in self.hosts:
119
            return self.hosts[host]
120
        elif 'default' in self.hosts:
121 122 123 124 125
            return self.hosts['default']
        else:
            return None

    def __repr__(self):
126
        """Dump the class data in the format of a .netrc file."""
127 128 129 130 131 132 133 134 135 136 137 138 139 140
        rep = ""
        for host in self.hosts.keys():
            attrs = self.hosts[host]
            rep = rep + "machine "+ host + "\n\tlogin " + repr(attrs[0]) + "\n"
            if attrs[1]:
                rep = rep + "account " + repr(attrs[1])
            rep = rep + "\tpassword " + repr(attrs[2]) + "\n"
        for macro in self.macros.keys():
            rep = rep + "macdef " + macro + "\n"
            for line in self.macros[macro]:
                rep = rep + line
            rep = rep + "\n"
        return rep

Tim Peters's avatar
Tim Peters committed
141
if __name__ == '__main__':
142
    print(netrc())