Passed
Push — master ( 1dcbd2...c58519 )
by Andreas
10:12
created

midcom_connection::_parse_url()   B

Complexity

Conditions 7
Paths 8

Size

Total Lines 31
Code Lines 22

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 19
CRAP Score 7.1243

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 7
eloc 22
nc 8
nop 3
dl 0
loc 31
ccs 19
cts 22
cp 0.8636
crap 7.1243
rs 8.6346
c 1
b 0
f 0
1
<?php
2
/**
3
 * @package midcom
4
 * @author CONTENT CONTROL http://www.contentcontrol-berlin.de/
5
 * @copyright CONTENT CONTROL http://www.contentcontrol-berlin.de/
6
 * @license http://www.gnu.org/licenses/lgpl.html GNU Lesser General Public License
7
 */
8
9
use midgard\portable\storage\connection;
10
use midgard\portable\api\error\exception as mgd_exception;
11
use midgard\portable\api\mgdobject;
12
13
/**
14
 * Wrapper for Midgard-related functionality
15
 *
16
 * @package midcom
17
 */
18
class midcom_connection
19
{
20
    private static array $_data = [];
21
22
    /**
23
     * DB connection setup routine
24
     *
25
     * @throws Exception We use regular exceptions here, because this might run before things are properly set up
26
     */
27
    public static function setup(string $basedir) : bool
28
    {
29
        if (file_exists($basedir . 'config/midgard-portable.inc.php')) {
30
            include $basedir . 'config/midgard-portable.inc.php';
31
            return midgard_connection::get_instance()->is_connected();
32
        }
33
        if (file_exists($basedir . 'config/midgard-portable-default.inc.php')) {
34
            include $basedir . 'config/midgard-portable-default.inc.php';
35
            //default config has in-memory db, so all the tables may be missing
36
            return midgard_storage::class_storage_exists('midgard_user');
37
        }
38
        throw new Exception("Could not connect to database, configuration file not found");
39
    }
40
41
    /**
42
     * Set Midgard log level
43
     *
44
     * @param string $loglevel Midgard log level
45
     */
46
    static function set_loglevel($loglevel) : bool
0 ignored issues
show
Best Practice introduced by
It is generally recommended to explicitly declare the visibility for methods.

Adding explicit visibility (private, protected, or public) is generally recommend to communicate to other developers how, and from where this method is intended to be used.

Loading history...
47
    {
48
        return midgard_connection::get_instance()->set_loglevel($loglevel);
49
    }
50
51
    /**
52
     * Set Midgard error code
53
     */
54 18
    public static function set_error(int $errorcode)
55
    {
56 18
        midgard_connection::get_instance()->set_error($errorcode);
57
    }
58
59
    /**
60
     * Get Midgard error code
61
     */
62 42
    public static function get_error() : int
63
    {
64 42
        return midgard_connection::get_instance()->get_error();
65
    }
66
67
    /**
68
     * Get Midgard error message
69
     */
70 57
    public static function get_error_string() : string
71
    {
72 57
        return midgard_connection::get_instance()->get_error_string();
73
    }
74
75
    /**
76
     * Perform a login against the midgard backend
77
     */
78 65
    public static function login(string $username, string $password, bool $trusted = false) : ?midgard_user
79
    {
80 65
        $login_tokens = [
81 65
            'login' => $username,
82 65
            'authtype' => midcom::get()->config->get('auth_type'),
83 65
        ];
84
85
        try {
86 65
            $user = new midgard_user($login_tokens);
87
        } catch (mgd_exception) {
88
            return null;
89
        }
90 65
        if (!$trusted && !self::verify_password($password, $user->password)) {
91
            return null;
92
        }
93
94 65
        $user->login();
95 65
        return $user;
96
    }
97
98 72
    public static function verify_password(string $password, string $hash) : bool
99
    {
100 72
        if (midcom::get()->config->get('auth_type') == 'Legacy') {
101 72
            return password_verify($password, $hash);
102
        }
103
        if (midcom::get()->config->get('auth_type') == 'SHA256') {
104
            $password = hash('sha256', $password);
105
        }
106
107
        return $password === $hash;
108
    }
109
110 90
    public static function prepare_password(string $password)
111
    {
112 90
        if (midcom::get()->config->get('auth_type') == 'Legacy') {
113 90
            return password_hash($password, PASSWORD_DEFAULT);
114
        }
115
        if (midcom::get()->config->get('auth_type') == 'SHA256') {
116
            return hash('sha256', $password);
117
        }
118
119
        return $password;
120
    }
121
122
    /**
123
     * Get current Midgard user ID
124
     */
125 61
    public static function get_user() : int
126
    {
127 61
        if ($user = midgard_connection::get_instance()->get_user()) {
128 33
            return $user->get_person()->id;
129
        }
130 28
        return 0;
131
    }
132
133
    /**
134
     * Logout the current user, if any
135
     */
136 1
    public static function logout()
137
    {
138 1
        if ($user = midgard_connection::get_instance()->get_user()) {
139 1
            $user->logout();
140
        }
141
    }
142
143 364
    private static function _get(string $key)
144
    {
145 364
        return self::$_data[$key] ?? null;
146
    }
147
148
    /**
149
     * Lists all available MgdSchema types
150
     */
151 6
    public static function get_schema_types() : array
152
    {
153 6
        if (!isset(self::$_data['schema_types'])) {
154
            $classnames = connection::get_em()->getConfiguration()->getMetadataDriverImpl()->getAllClassNames();
155
156
            self::$_data['schema_types'] = array_filter($classnames, function(string $input) {
157
                return is_subclass_of($input, mgdobject::class);
158
            });
159
        }
160 6
        return self::$_data['schema_types'];
161
    }
162
163
    /**
164
     * Get various pieces of information extracted from the URL
165
     *
166
     * @return mixed The data for the requested key or null if it doesn't exist
167
     */
168 364
    public static function get_url(string $key)
169
    {
170 364
        static $parsed = false;
171 364
        if (!$parsed) {
172
            $self = defined('OPENPSA2_PREFIX') ? OPENPSA2_PREFIX : '/';
173
174
            // we're only interested in the path, so use a dummy domain for simplicity's sake
175
            $url_path = (string) parse_url("https://openpsa2.org{$_SERVER['REQUEST_URI']}", PHP_URL_PATH);
176
            if ($self !== '/') {
177
                $url_path = preg_replace('|^' . $self . '|', '/', $url_path);
178
            }
179
            self::_parse_url($url_path, $self, substr($self, 0, -1));
180
            $parsed = true;
181
        }
182
183 364
        return self::_get($key);
184
    }
185
186
    /**
187
     * Enables themes to have subdirectories (which have a similar effect to mgd1 pages)
188
     *
189
     * @param string $uri The request path
190
     * @param string $self The instance's root URL
191
     * @param string $prefix The root URL's prefix, if any (corresponds to mgd1 host)
192
     */
193 1
    private static function _parse_url(string $uri, string $self, string $prefix)
194
    {
195 1
        $uri = preg_replace('/\/[\/]+/i', '/', $uri);
196 1
        $path_parts = explode('/', $uri);
197 1
        $page_style = '';
198 1
        $path = $self;
199
200 1
        $args_started = false;
201 1
        foreach ($path_parts as $part) {
202 1
            if ($part === '') {
203 1
                continue;
204
            }
205 1
            if (    midcom::get()->config->get('theme')
206 1
                 && !$args_started
207 1
                 && self::check_page_exists($part)) {
208
                $page_style .= '/' . $part;
209
                $self .= $part . '/';
210
            } else {
211 1
                $path .= $part . '/';
212 1
                $args_started = true;
213
            }
214
        }
215
216 1
        if (!str_ends_with($uri, '/')) {
217
            $path = substr($path, 0, -1);
218
        }
219
220 1
        self::$_data['page_style'] = $page_style;
221 1
        self::$_data['uri'] = $path;
222 1
        self::$_data['self'] = $self;
223 1
        self::$_data['prefix'] = $prefix;
224
    }
225
226
    /**
227
     * Iterate through possible page directories in style-tree
228
     * and check if the page exists (as a folder).
229
     */
230 1
    private static function check_page_exists(string $page_name) : bool
231
    {
232 1
        $prefix = midcom::get()->getProjectDir() . '/var/themes/';
233 1
        $path_array = explode('/', midcom::get()->config->get('theme'));
0 ignored issues
show
Bug introduced by
It seems like midcom::get()->config->get('theme') can also be of type null; however, parameter $string of explode() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

233
        $path_array = explode('/', /** @scrutinizer ignore-type */ midcom::get()->config->get('theme'));
Loading history...
234 1
        while (!empty($path_array)) {
235 1
            $theme_path = implode('/', $path_array);
236 1
            if (is_dir($prefix . $theme_path . '/style/' . $page_name)) {
237
                return true;
238
            }
239 1
            array_pop($path_array);
240
        }
241 1
        return false;
242
    }
243
}
244