Key   A
last analyzed

Complexity

Total Complexity 14

Size/Duplication

Total Lines 138
Duplicated Lines 0 %

Test Coverage

Coverage 100%

Importance

Changes 2
Bugs 0 Features 0
Metric Value
eloc 33
dl 0
loc 138
ccs 17
cts 17
cp 1
rs 10
c 2
b 0
f 0
wmc 14

6 Methods

Rating   Name   Duplication   Size   Complexity  
A build() 0 9 2
A getOrdinal() 0 7 4
A parse() 0 3 1
A parseNoun() 0 11 4
A splitPossessions() 0 3 2
A isPossessive() 0 3 1
1
<?php namespace Chekote\NounStore;
2
3
use InvalidArgumentException;
4
5
class Key
6
{
7
    use Singleton;
8
9
    const ORDINAL_ST = 'st';
10
    const ORDINAL_ND = 'nd';
11
    const ORDINAL_RD = 'rd';
12
    const ORDINAL_TH = 'th';
13
14
    protected static $ordinals = [
15
        0 => self::ORDINAL_TH,
16
        1 => self::ORDINAL_ST,
17
        2 => self::ORDINAL_ND,
18
        3 => self::ORDINAL_RD,
19
        4 => self::ORDINAL_TH,
20
        5 => self::ORDINAL_TH,
21
        6 => self::ORDINAL_TH,
22
        7 => self::ORDINAL_TH,
23
        8 => self::ORDINAL_TH,
24
        9 => self::ORDINAL_TH,
25
    ];
26
27
    const POSSESSION = "'s ";
28
29
    /**
30
     * Builds a key from it's separate key and index values.
31
     *
32
     * @example buildKey("Item", null): "Item"
33
     * @example buildKey("Item", 0): "1st Item"
34
     * @example buildKey("Item", 1): "2nd Item"
35
     * @example buildKey("Item", 2): "3rd Item"
36
     *
37
     * @param  string                   $key   The key to check.
38
     * @param  int|null                 $index The index (zero indexed) value for the key. If not specified, the method
39
     *                                         will not add an index notation to the key.
40
     * @throws InvalidArgumentException if $index is less than -1. Note: It should really be zero or higher, but this
41
     *                                  method does not assert that. The error is bubbling up from getOrdinal()
42 3
     * @return string                   the key with the index, or just the key if index is null.
43
     */
44 3
    public function build($key, $index): string
45 1
    {
46
        if ($index === null) {
47
            return $key;
48 2
        }
49
50 2
        $nth = $index + 1;
51
52
        return $nth . $this->getOrdinal($nth) . ' ' . $key;
53
    }
54
55
    /**
56
     * Provides the ordinal notation for the specified nth number.
57
     *
58
     * @param  int                      $nth the number to determine the ordinal for
59
     * @throws InvalidArgumentException if $nth is not a positive number.
60 24
     * @return string                   the ordinal
61
     */
62 24
    public function getOrdinal($nth): string
63 1
    {
64
        if ($nth < 0) {
65
            throw new InvalidArgumentException('$nth must be a positive number');
66 23
        }
67
68
        return $nth > 9 && $nth < 20 ? self::ORDINAL_TH : self::$ordinals[substr($nth, -1)];
69
    }
70
71
    /**
72
     * Parses a key into its individual nouns and indexes.
73
     *
74
     * @example parseNoun("Customer"): [["Customer", null]]
75
     * @example parseNoun("2nd Customer"): [["Customer", 1]]
76
     * @example parseNoun("Customer's Car"): [["Customer", null], ["Car", null]]
77
     * @example parseNoun("2nd Customer's Car"): [["Customer", 1], ["Car", null]]
78
     * @example parseNoun("4th Customer's 3rd Car"): [["Customer", 3], ["Car", 2]]
79
     *
80
     * @param  string                   $key the key to parse.
81
     * @throws InvalidArgumentException if the key syntax is invalid.
82
     * @return array[]                  a array of tuples. With each tuple have the noun with the nth removed as the
83
     *                                  1st item, and the index that the nth translates to as the 2nd or null if no
84
     *                                  nth was specified.
85 14
     */
86
    public function parse($key): array
87 14
    {
88 11
        return array_map([$this, 'parseNoun'], $this->splitPossessions($key));
89 5
    }
90 5
91
    /**
92
     * Parses a noun into the separate key and index value.
93
     *
94 6
     * @example parseNoun("Item"): ["Item", null]
95 6
     * @example parseNoun("1st Item"): ["Item", 0]
96
     * @example parseNoun("2nd Item"): ["Item", 1]
97
     * @example parseNoun("3rd Item"): ["Item", 2]
98 9
     *
99
     * @param  string                   $noun the key to parse.
100
     * @throws InvalidArgumentException if the key syntax is invalid.
101
     * @return array                    a tuple, the 1st being the key with the nth removed, and the 2nd being the
102
     *                                  index that the nth translates to, or null if no nth was specified.
103
     */
104
    protected function parseNoun($noun): array
105
    {
106
        if (!preg_match("/^(([1-9]\d*)(?:st|nd|rd|th) )?([^']+)$/", $noun, $matches)) {
107
            throw new InvalidArgumentException('Key syntax is invalid');
108
        }
109
110
        // @todo use null coalescing operator when upgrading to PHP 7
111
        $index = isset($matches[2]) && $matches[2] !== '' ? $matches[2] - 1 : null;
112
        $noun = $matches[3];
113
114
        return [$noun, $index];
115
    }
116
117
    /**
118
     * Determines if the specified key is a possessive noun.
119
     *
120
     * @param  string $key
121
     * @return bool   true if the key is possessive, false if not
122
     */
123
    protected function isPossessive($key): bool
124
    {
125
        return strpos($key, self::POSSESSION) !== false;
126
    }
127
128
    /**
129
     * Splits a possessive key into its separate nouns.
130
     *
131
     * @example splitPossessions("Customer's Car"): ['Customer', 'Car']
132
     * @example splitPossessions("8th Customer's Car"): ['8th Customer', 'Car']
133
     * @example splitPossessions("Customer's 2nd Car"): ['Customer', '2nd Car']
134
     * @example splitPossessions("7th Customer's 4th Car"): ['7th Customer', '4th Car']
135
     * @example splitPossessions("7th Customer's 4th Car's 2nd Wheel"): ['7th Customer', '4th Car', '2nd Wheel']
136
     *
137
     * @param  string   $key the possessive key to parse
138
     * @return string[] an array of nouns
139
     */
140
    protected function splitPossessions(string $key): array
141
    {
142
        return ($nouns = preg_split('/' . self::POSSESSION . '/', $key)) ? $nouns : [];
143
    }
144
}
145