InflectorTest.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. <?php
  2. /**
  3. * Lithium: the most rad php framework
  4. *
  5. * @copyright Copyright 2013, Union of RAD (http://union-of-rad.org)
  6. * @license http://opensource.org/licenses/bsd-license.php The BSD License
  7. */
  8. namespace lithium\tests\cases\util;
  9. use lithium\util\Inflector;
  10. use lithium\analysis\Inspector;
  11. class InflectorTest extends \lithium\test\Unit {
  12. public static $_test = 'bar';
  13. public function tearDown() {
  14. Inflector::reset();
  15. }
  16. /**
  17. * Tests singularization inflection rules
  18. *
  19. * @return void
  20. */
  21. public function testSingularize() {
  22. $this->assertEqual(Inflector::singularize('categorias'), 'categoria');
  23. $this->assertEqual(Inflector::singularize('menus'), 'menu');
  24. $this->assertEqual(Inflector::singularize('news'), 'news');
  25. $this->assertEqual(Inflector::singularize('food_menus'), 'food_menu');
  26. $this->assertEqual(Inflector::singularize('Menus'), 'Menu');
  27. $this->assertEqual(Inflector::singularize('FoodMenus'), 'FoodMenu');
  28. $this->assertEqual(Inflector::singularize('houses'), 'house');
  29. $this->assertEqual(Inflector::singularize('powerhouses'), 'powerhouse');
  30. $this->assertEqual(Inflector::singularize('quizzes'), 'quiz');
  31. $this->assertEqual(Inflector::singularize('Buses'), 'Bus');
  32. $this->assertEqual(Inflector::singularize('buses'), 'bus');
  33. $this->assertEqual(Inflector::singularize('matrix_rows'), 'matrix_row');
  34. $this->assertEqual(Inflector::singularize('matrices'), 'matrix');
  35. $this->assertEqual(Inflector::singularize('vertices'), 'vertex');
  36. $this->assertEqual(Inflector::singularize('indices'), 'index');
  37. $this->assertEqual(Inflector::singularize('Aliases'), 'Alias');
  38. $this->assertEqual(Inflector::singularize('Alias'), 'Alias');
  39. $this->assertEqual(Inflector::singularize('Media'), 'Media');
  40. $this->assertEqual(Inflector::singularize('alumni'), 'alumnus');
  41. $this->assertEqual(Inflector::singularize('bacilli'), 'bacillus');
  42. $this->assertEqual(Inflector::singularize('cacti'), 'cactus');
  43. $this->assertEqual(Inflector::singularize('foci'), 'focus');
  44. $this->assertEqual(Inflector::singularize('fungi'), 'fungus');
  45. $this->assertEqual(Inflector::singularize('nuclei'), 'nucleus');
  46. $this->assertEqual(Inflector::singularize('octopuses'), 'octopus');
  47. $this->assertEqual(Inflector::singularize('radii'), 'radius');
  48. $this->assertEqual(Inflector::singularize('stimuli'), 'stimulus');
  49. $this->assertEqual(Inflector::singularize('syllabi'), 'syllabus');
  50. $this->assertEqual(Inflector::singularize('termini'), 'terminus');
  51. $this->assertEqual(Inflector::singularize('viri'), 'virus');
  52. $this->assertEqual(Inflector::singularize('people'), 'person');
  53. $this->assertEqual(Inflector::singularize('gloves'), 'glove');
  54. $this->assertEqual(Inflector::singularize('doves'), 'dove');
  55. $this->assertEqual(Inflector::singularize('lives'), 'life');
  56. $this->assertEqual(Inflector::singularize('leaves'), 'leaf');
  57. $this->assertEqual(Inflector::singularize('knives'), 'knife');
  58. $this->assertEqual(Inflector::singularize('wolves'), 'wolf');
  59. $this->assertEqual(Inflector::singularize('shelves'), 'shelf');
  60. $this->assertEqual(Inflector::singularize('causes'), 'cause');
  61. $this->assertEqual(Inflector::singularize(''), '');
  62. }
  63. /**
  64. * Tests pluralization inflection rules
  65. *
  66. * @return void
  67. */
  68. public function testPluralize() {
  69. $this->assertEqual(Inflector::pluralize('categoria'), 'categorias');
  70. $this->assertEqual(Inflector::pluralize('house'), 'houses');
  71. $this->assertEqual(Inflector::pluralize('powerhouse'), 'powerhouses');
  72. $this->assertEqual(Inflector::pluralize('Bus'), 'Buses');
  73. $this->assertEqual(Inflector::pluralize('bus'), 'buses');
  74. $this->assertEqual(Inflector::pluralize('menu'), 'menus');
  75. $this->assertEqual(Inflector::pluralize('news'), 'news');
  76. $this->assertEqual(Inflector::pluralize('food_menu'), 'food_menus');
  77. $this->assertEqual(Inflector::pluralize('Menu'), 'Menus');
  78. $this->assertEqual(Inflector::pluralize('FoodMenu'), 'FoodMenus');
  79. $this->assertEqual(Inflector::pluralize('quiz'), 'quizzes');
  80. $this->assertEqual(Inflector::pluralize('matrix_row'), 'matrix_rows');
  81. $this->assertEqual(Inflector::pluralize('matrix'), 'matrices');
  82. $this->assertEqual(Inflector::pluralize('vertex'), 'vertices');
  83. $this->assertEqual(Inflector::pluralize('index'), 'indices');
  84. $this->assertEqual(Inflector::pluralize('Alias'), 'Aliases');
  85. $this->assertEqual(Inflector::pluralize('Aliases'), 'Aliases');
  86. $this->assertEqual(Inflector::pluralize('Media'), 'Media');
  87. $this->assertEqual(Inflector::pluralize('alumnus'), 'alumni');
  88. $this->assertEqual(Inflector::pluralize('bacillus'), 'bacilli');
  89. $this->assertEqual(Inflector::pluralize('cactus'), 'cacti');
  90. $this->assertEqual(Inflector::pluralize('focus'), 'foci');
  91. $this->assertEqual(Inflector::pluralize('fungus'), 'fungi');
  92. $this->assertEqual(Inflector::pluralize('nucleus'), 'nuclei');
  93. $this->assertEqual(Inflector::pluralize('octopus'), 'octopuses');
  94. $this->assertEqual(Inflector::pluralize('radius'), 'radii');
  95. $this->assertEqual(Inflector::pluralize('stimulus'), 'stimuli');
  96. $this->assertEqual(Inflector::pluralize('syllabus'), 'syllabi');
  97. $this->assertEqual(Inflector::pluralize('terminus'), 'termini');
  98. $this->assertEqual(Inflector::pluralize('virus'), 'viri');
  99. $this->assertEqual(Inflector::pluralize('person'), 'people');
  100. $this->assertEqual(Inflector::pluralize('people'), 'people');
  101. $this->assertEqual(Inflector::pluralize('glove'), 'gloves');
  102. $this->assertEqual(Inflector::pluralize('leaf'), 'leaves');
  103. $this->assertEqual(Inflector::pluralize('ContactPeople'), 'ContactPeople');
  104. $this->assertEqual(Inflector::pluralize(''), '');
  105. $result = Inflector::pluralize('errata');
  106. $this->assertNull(Inflector::rules('plural', array('/rata/' => '\1ratum')));
  107. $this->assertEqual(Inflector::pluralize('errata'), $result);
  108. Inflector::reset();
  109. $this->assertNotEqual(Inflector::pluralize('errata'), $result);
  110. }
  111. /**
  112. * testInflectorSlug method
  113. *
  114. * @return void
  115. */
  116. public function testSlug() {
  117. $result = Inflector::slug('Foo Bar: Not just for breakfast any-more');
  118. $expected = 'Foo-Bar-Not-just-for-breakfast-any-more';
  119. $this->assertEqual($expected, $result);
  120. $result = Inflector::slug('Foo Bar: Not just for breakfast any-more', '_');
  121. $expected = 'Foo_Bar_Not_just_for_breakfast_any_more';
  122. $this->assertEqual($expected, $result);
  123. $result = Inflector::slug('this/is/a/path', '_');
  124. $expected = 'this_is_a_path';
  125. $this->assertEqual($expected, $result);
  126. $result = Inflector::slug('Foo Bar: Not just for breakfast any-more', "+");
  127. $expected = 'Foo+Bar+Not+just+for+breakfast+any+more';
  128. $this->assertEqual($expected, $result);
  129. $result = Inflector::slug('Äpfel Über Öl grün ärgert groß öko');
  130. $expected = 'Aepfel-Ueber-Oel-gruen-aergert-gross-oeko';
  131. $this->assertEqual($expected, $result);
  132. $result = Inflector::slug('The truth - and- more- news');
  133. $expected = 'The-truth-and-more-news';
  134. $this->assertEqual($expected, $result);
  135. $result = Inflector::slug('The truth: and more news');
  136. $expected = 'The-truth-and-more-news';
  137. $this->assertEqual($expected, $result);
  138. $message = 'La langue française est un attribut de souveraineté en France';
  139. $result = Inflector::slug($message, '-');
  140. $expected = 'La-langue-francaise-est-un-attribut-de-souverainete-en-France';
  141. $this->assertEqual($expected, $result);
  142. $result = Inflector::slug('!@$#exciting stuff! - what !@-# was that?');
  143. $expected = 'exciting-stuff-what-was-that';
  144. $this->assertEqual($expected, $result);
  145. $result = Inflector::slug('20% of profits went to me!');
  146. $expected = '20-of-profits-went-to-me';
  147. $this->assertEqual($expected, $result);
  148. $result = Inflector::slug('#this melts your face1#2#3');
  149. $expected = 'this-melts-your-face1-2-3';
  150. $this->assertEqual($expected, $result);
  151. $result = Inflector::slug('ThisMeltsYourFace');
  152. $expected = 'This-Melts-Your-Face';
  153. $this->assertEqual($expected, $result);
  154. $result = Inflector::slug('DŽip Džip džip');
  155. $expected = 'Dzip-Dzip-dzip';
  156. $this->assertEqual($expected, $result);
  157. $result = Inflector::slug('Šuma šuma');
  158. $expected = 'Suma-suma';
  159. $this->assertEqual($expected, $result);
  160. $result = Inflector::slug('Čamac čamac');
  161. $expected = 'Camac-camac';
  162. $this->assertEqual($expected, $result);
  163. $result = Inflector::slug('Ćasa ćasa');
  164. $expected = 'Casa-casa';
  165. $this->assertEqual($expected, $result);
  166. $result = Inflector::slug('Život život');
  167. $expected = 'Zivot-zivot';
  168. $this->assertEqual($expected, $result);
  169. $result = Inflector::slug('Đorđe đorđe');
  170. $expected = 'Djordje-djordje';
  171. $this->assertEqual($expected, $result);
  172. }
  173. public function testAddingInvalidRules() {
  174. $before = array(
  175. Inflector::rules('singular'),
  176. Inflector::rules('plural'),
  177. Inflector::rules('transliteration')
  178. );
  179. $this->assertNull(Inflector::rules('foo'));
  180. $this->assertIdentical($before, array(
  181. Inflector::rules('singular'),
  182. Inflector::rules('plural'),
  183. Inflector::rules('transliteration')
  184. ));
  185. }
  186. public function testAddingSingularizationRules() {
  187. $before = Inflector::rules('singular');
  188. $result = Inflector::singularize('errata');
  189. $this->assertNull(Inflector::rules('singular', array('/rata/' => '\1ratus')));
  190. $this->assertEqual(Inflector::singularize('errata'), $result);
  191. Inflector::reset();
  192. $this->assertNotEqual(Inflector::singularize('errata'), $result);
  193. $after = Inflector::rules('singular');
  194. $expected = array(
  195. 'rules', 'irregular', 'uninflected', 'regexUninflected', 'regexIrregular'
  196. );
  197. $this->assertEqual(array_keys($before), $expected);
  198. $this->assertEqual(array_keys($after), $expected);
  199. $result = array_diff($after['rules'], $before['rules']);
  200. $this->assertEqual($result, array('/rata/' => '\1ratus'));
  201. foreach (array('irregular', 'uninflected', 'regexUninflected', 'regexIrregular') as $key) {
  202. $this->assertIdentical($before[$key], $after[$key]);
  203. }
  204. $this->assertNull(Inflector::rules('singular', array('rules' => array(
  205. '/rata/' => '\1ratus'
  206. ))));
  207. $this->assertIdentical(Inflector::rules('singular'), $after);
  208. }
  209. /**
  210. * Tests that rules for uninflected singular words are kept in sync with the plural, and vice
  211. * versa.
  212. *
  213. * @return void
  214. */
  215. public function testIrregularWords() {
  216. $expectedPlural = Inflector::rules('plural');
  217. $this->assertFalse(isset($expectedPlural['irregular']['bar']));
  218. $expectedSingular = Inflector::rules('singular');
  219. $this->assertFalse(isset($expectedSingular['irregular']['foo']));
  220. Inflector::rules('singular', array('irregular' => array('foo' => 'bar')));
  221. $resultSingular = Inflector::rules('singular');
  222. $this->assertEqual($resultSingular['irregular']['foo'], 'bar');
  223. unset($resultSingular['irregular']['foo']);
  224. $this->assertEqual($resultSingular, $expectedSingular);
  225. $resultPlural = Inflector::rules('plural');
  226. $this->assertEqual($resultPlural['irregular']['bar'], 'foo');
  227. unset($resultPlural['irregular']['bar']);
  228. $this->assertEqual($resultPlural, $expectedPlural);
  229. }
  230. /**
  231. * testVariableNaming method
  232. *
  233. * @return void
  234. */
  235. public function testCamelize() {
  236. $this->assertEqual(Inflector::camelize('test-field'), 'TestField');
  237. $this->assertEqual(Inflector::camelize('test_field'), 'TestField');
  238. $this->assertEqual(Inflector::camelize('test_fieLd', false), 'testFieLd');
  239. $this->assertEqual(Inflector::camelize('test field', false), 'testField');
  240. $this->assertEqual(Inflector::camelize('Test_field', false), 'testField');
  241. }
  242. /**
  243. * testClassNaming method
  244. *
  245. * @return void
  246. */
  247. public function testClassify() {
  248. $this->assertEqual(Inflector::classify('artists_genres'), 'ArtistsGenre');
  249. $this->assertEqual(Inflector::classify('file_systems'), 'FileSystem');
  250. $this->assertEqual(Inflector::classify('news'), 'News');
  251. }
  252. /**
  253. * testTableNaming method
  254. *
  255. * @return void
  256. */
  257. public function testTabelize() {
  258. $this->assertEqual(Inflector::tableize('ArtistsGenre'), 'artists_genres');
  259. $this->assertEqual(Inflector::tableize('FileSystem'), 'file_systems');
  260. $this->assertEqual(Inflector::tableize('News'), 'news');
  261. }
  262. /**
  263. * testHumanization method
  264. *
  265. * @return void
  266. */
  267. public function testHumanize() {
  268. $this->assertEqual(Inflector::humanize('posts'), 'Posts');
  269. $this->assertEqual(Inflector::humanize('posts_tags'), 'Posts Tags');
  270. $this->assertEqual(Inflector::humanize('file_systems'), 'File Systems');
  271. $this->assertEqual(Inflector::humanize('the-post-title', '-'), 'The Post Title');
  272. }
  273. /**
  274. * Tests adding transliterated characters to the map used in `Inflector::slug()`.
  275. *
  276. * @return void
  277. */
  278. public function testAddTransliterations() {
  279. $this->assertEqual(Inflector::slug('Montréal'), 'Montreal');
  280. $this->assertNotEqual(Inflector::slug('Écaussines'), 'Ecaussines');
  281. Inflector::rules('transliteration', array('/É|Ê/' => 'E'));
  282. $this->assertEqual(Inflector::slug('Écaussines-d\'Enghien'), 'Ecaussines-d-Enghien');
  283. $this->assertNotEqual(Inflector::slug('JØRGEN'), 'JORGEN');
  284. Inflector::rules('transliteration', array('/Ø/' => 'O'));
  285. $this->assertEqual(Inflector::slug('JØRGEN'), 'JORGEN');
  286. $this->assertNotEqual(Inflector::slug('ÎÍ'), 'II');
  287. Inflector::rules('transliteration', array('/Î|Í/' => 'I'));
  288. $this->assertEqual(Inflector::slug('ÎÍ'), 'II');
  289. $this->assertEqual(Inflector::slug('ABc'), 'ABc');
  290. Inflector::rules('transliteration', array('AB' => 'a'));
  291. $this->assertEqual(Inflector::slug('ABc'), 'aac');
  292. }
  293. public function testAddingUninflectedWords() {
  294. $this->assertEqual(Inflector::pluralize('bord'), 'bords');
  295. Inflector::rules('uninflected', 'bord');
  296. $this->assertEqual(Inflector::pluralize('bord'), 'bord');
  297. }
  298. /**
  299. * Tests the storage mechanism for `$_underscored`, `$_camelized`,
  300. * `$_humanized` and `$_pluralized`.
  301. *
  302. * @return void
  303. */
  304. public function testStorageMechanism() {
  305. Inflector::reset();
  306. $expected = array('TestField' => 'test_field');
  307. $this->assertFalse($this->_getProtectedValue('$_underscored'));
  308. $this->assertEqual(Inflector::underscore('TestField'), 'test_field');
  309. $this->assertEqual($expected, $this->_getProtectedValue('$_underscored'));
  310. $this->assertEqual(Inflector::underscore('TestField'), 'test_field');
  311. $expected = array('test_field' => 'TestField');
  312. $this->assertFalse($this->_getProtectedValue('$_camelized'));
  313. $this->assertEqual(Inflector::camelize('test_field', true), 'TestField');
  314. $this->assertEqual($expected, $this->_getProtectedValue('$_camelized'));
  315. $this->assertEqual(Inflector::camelize('test_field', true), 'TestField');
  316. $expected = array('test_field:_' => 'Test Field');
  317. $this->assertFalse($this->_getProtectedValue('$_humanized'));
  318. $this->assertEqual(Inflector::humanize('test_field'), 'Test Field');
  319. $this->assertEqual($expected, $this->_getProtectedValue('$_humanized'));
  320. $this->assertEqual(Inflector::humanize('test_field'), 'Test Field');
  321. $expected = array('field' => 'fields');
  322. $this->assertFalse($this->_getProtectedValue('$_pluralized'));
  323. $this->assertEqual(Inflector::pluralize('field'), 'fields');
  324. $this->assertEqual($expected, $this->_getProtectedValue('$_pluralized'));
  325. $this->assertEqual(Inflector::pluralize('field'), 'fields');
  326. }
  327. /**
  328. * This is a helper method for testStorageMechanism to fetch a private
  329. * property of the Inflector class.
  330. *
  331. * @param string $property
  332. * @return string The value of the property.
  333. */
  334. protected function _getProtectedValue($property) {
  335. $info = Inspector::info("lithium\util\Inflector::{$property}");
  336. return $info['value'];
  337. }
  338. }
  339. ?>