Completed
Pull Request — master (#442)
by
unknown
02:14
created

Auth   A

Complexity

Total Complexity 11

Size/Duplication

Total Lines 70
Duplicated Lines 0 %
Metric Value
dl 0
loc 70
rs 10
wmc 11

1 Method

Rating   Name   Duplication   Size   Complexity  
F run() 0 69 11
1
#!/usr/bin/env python
2
3
# Licensed to the StackStorm, Inc ('StackStorm') under one or more
4
# contributor license agreements.  See the NOTICE file distributed with
5
# this work for additional information regarding copyright ownership.
6
# The ASF licenses this file to You under the Apache License, Version 2.0
7
# (the "License"); you may not use this file except in compliance with
8
# the License.  You may obtain a copy of the License at
9
#
10
#     http://www.apache.org/licenses/LICENSE-2.0
11
#
12
# Unless required by applicable law or agreed to in writing, software
13
# distributed under the License is distributed on an "AS IS" BASIS,
14
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15
# See the License for the specific language governing permissions and
16
# limitations under the License.
17
18
import urllib
19
20
from st2actions.runners.pythonrunner import Action
21
import duo_client
22
23
24
class Auth(Action):
25
    def run(self, username, factor,
26
            ipaddr, device, push_type, passcode, pushinfo):
27
        """
28
        Auth against the Duo Platorm.
29
30
        Returns: An dict with info returned by Duo.
31
32
        Raises:
33
          ValueError: 'Duo config not found in config' or 'Invalid factor'
34
          RuntimeError: 'Failed auth.'
35
        """
36
37
        try:
38
            ikey = self.config['auth']['ikey']
39
            skey = self.config['auth']['skey']
40
            host = self.config['auth']['host']
41
        except KeyError:
42
            raise ValueError("Duo config not found in config.")
43
44
        auth = duo_client.Auth(ikey=ikey,
45
                               skey=skey,
46
                               host=host)
47
48
        auth_kargs = {}
49
50
        if factor == "auto" or factor == "push":
51
            auth_kargs['type'] = push_type
52
            auth_kargs['device'] = device
53
54
            if ipaddr is not None:
55
                auth_kargs['ipaddr'] = ipaddr
56
57
            if pushinfo is not None:
58
                info = {}
59
                for value in pushinfo.split('; '):
60
                    (key, value) = value.split('=')
61
                    info[key] = value
62
63
                encoded = urllib.urlencode(info)
64
                auth_kargs['pushinfo'] = encoded
65
        elif factor == "passcode":
66
            auth_kargs['passcode'] = passcode
67
        elif factor == "phone":
68
            auth_kargs['device'] = device
69
        elif factor == "sms":
70
            # As 'sms' just denies and then we do not support it
71
            # requires re-authentication.
72
73
            print "Denied, we do not support SMS!"
74
            raise ValueError("Denied, we do not support SMS!")
75
        else:
76
            raise ValueError("Invalid factor!")
77
78
        try:
79
            data = auth.auth(factor=factor,
80
                             username=username,
81
                             **auth_kargs)
82
        except RuntimeError, e:
83
            print "Error: %s" % e
84
            raise RuntimeError("Error: %s" % e)
85
        else:
86
            if data['result'] == "allow":
87
                return data
88
            elif data['result'] == "deny":
89
                print data['status_msg']
90
                raise RuntimeError("{}".format(
91
                    data['status_msg']))
92
            else:
93
                raise RuntimeError("Invalid status")
94