You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

272 lines
10 KiB

10 years ago
10 years ago
10 years ago
9 years ago
9 years ago
9 years ago
9 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
8 years ago
9 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
  1. ![Tom Lingham Laravel Searchy](http://tomlingham.com/github/header-searchy.png)
  2. ## Laravel 5+ Searchy
  3. ### Database Searching Made Easy
  4. Searchy is an; easy-to-use, light-weight, MySQL only, Laravel package that makes running user driven searches on data in your models simple and effective.
  5. It uses pseudo fuzzy searching and other weighted mechanics depending on the search driver that you have enabled.
  6. It requires no other software installed on your server (so can be a little slower than dedicated search programs) but can be set up and ready to go in minutes.
  7. #### !! Laravel 4 !!
  8. Looking for Laravel 4 compatible Searchy? Checkout the 1.0 branch :)
  9. https://github.com/TomLingham/Laravel-Searchy/tree/1.0
  10. ## Installation
  11. Add `"tom-lingham/searchy" : "2.*"` to your composer.json file under `require`:
  12. ```json
  13. "require": {
  14. "laravel/framework": "5.*",
  15. "tom-lingham/searchy" : "2.*"
  16. }
  17. ```
  18. Run `composer update` in your terminal to pull down the package into your vendors folder.
  19. Add the service provider to the `providers` array in Laravel's `./config/app.php` file:
  20. ```php
  21. TomLingham\Searchy\SearchyServiceProvider::class
  22. ```
  23. Add the Alias to the `aliases` array in Laravel's `./config/app.php` file if you want to have quick access to it in your application:
  24. ```php
  25. 'Searchy' => TomLingham\Searchy\Facades\Searchy::class
  26. ```
  27. ## Usage
  28. To use Searchy, you can take advantage of magic methods.
  29. If you are searching the name and email column/field of users in a `users` table you would, for example run:
  30. ```php
  31. $users = Searchy::users('name', 'email')->query('John Smith')->get();
  32. ```
  33. You can also write this as:
  34. ```php
  35. $users = Searchy::search('users')->fields('name', 'email')->query('John Smith')->get();
  36. ```
  37. In this case, pass the columns you want to search through to the `fields()` method.
  38. These examples both return an array of Objects containing your search results. You can use `getQuery()` instead of `get()` to return an instance of the Database Query Object in case you want to do further manipulation to the results:
  39. ```php
  40. $users = Searchy::search('users')->fields('name', 'email')->query('John Smith')
  41. ->getQuery()->having('relevance', '>', 20)->get();
  42. ```
  43. ### Limit on results returned
  44. To limit your results, you can use Laravel's built in DatabaseQuery Builder method and chain further methods to narrow your results.
  45. ```php
  46. // Only get the top 10 results
  47. $users = Searchy::search('users')->fields('name', 'email')->query('John Smith')
  48. ->getQuery()->limit(10)->get();
  49. ```
  50. ### Searching multiple Columns
  51. You can also add multiple arguments to the list of fields/columns to search by.
  52. For example, if you want to search the name, email address and username of a user, you might run:
  53. ```php
  54. $users = Searchy::users('name', 'email', 'username')->query('John Smith')->get();
  55. ```
  56. If you need to build your table list dynamically, you can also pass an array of fields instead as the first argument (All other following arguments will be ignored):
  57. ```php
  58. $users = Searchy::users(['name', 'email', 'username'])->query('John Smith')->get();
  59. ```
  60. ### Searching Joined/Concatenated Columns
  61. Sometimes you may want to leverage searches on concatenated column. For example, on a `first_name` and `last_name` field but you only want to run the one query. To do this can separate columns with a double colon:
  62. ```php
  63. $users = Searchy::users('first_name::last_name')->query('John Smith')->get();
  64. ```
  65. ### Soft Deleted Records
  66. By default soft deletes will not be included in your results. However, if you wish to include soft deleted records you can do so by adding the `withTrashed()` after specifying your table and fields;
  67. ```php
  68. Searchy::users('name')->withTrashed()->query('Batman')->get();
  69. ```
  70. ### Return only specific columns
  71. You can specify which columns to return in your search:
  72. ```php
  73. $users = Searchy::users('first_name::last_name')->query('John Smith')->select('first_name')->get();
  74. // Or you can swap those around...
  75. $users = Searchy::users('first_name::last_name')->select('first_name')->query('John Smith')->get();
  76. ```
  77. This will, however, also return the `relevance` aliased column regardless of what is entered here.
  78. ## How to get a Laravel Eloquent Collection
  79. Transforming the search results into a collection of Laravel Eloquent models is outside the scope of this project. However, an easy way to achieve this without hitting your database more than necessary is to use the Eloquent `hydrate()` method.
  80. ```php
  81. \App\User::hydrate(Searchy::users('name', 'email')->query('Andrew')->get());
  82. ```
  83. This method creates a collection of models from a plain arrays. This is just our case because Searchy results are provided as arrays, and using `hydrate` we will converted them to instances of `User` model.
  84. ## Unicode Characters Support
  85. If you are having issues with the returned results because you have unicode characters in your search data, you can use the `FuzzySearchUnicodeDriver`.
  86. _PLEASE NOTE: There is no sanitization of strings passed through to the `FuzzySearchUnicodeDriver` prior to inserting into raw MySQL statements. You will have to sanitize the string yourself first or risk opening up your application to SQL injection attacks. You have been warned._
  87. To use, first follow the instructions to publish your configuration file (`php artisan vendor:publish`) and change your default driver from `fuzzy` to `ufuzzy`.
  88. ```php
  89. return [
  90. 'default' => 'ufuzzy',
  91. ...
  92. ]
  93. ```
  94. ## Configuration
  95. You can publish the configuration file to your `app` directory and override the settings by running `php artisan vendor:publish` to copy the configuration to your config folder as `searchy.php`
  96. You can set the default driver to use for searches in the configuration file. Your options (at this stage) are: `fuzzy`, `simple` and `levenshtein`.
  97. You can also override these methods using the following syntax when running a search:
  98. ```php
  99. Searchy::driver('fuzzy')->users('name')->query('Batman')->get();
  100. ```
  101. ## Drivers
  102. Searchy takes advantage of 'Drivers' to handle matching various conditions of the fields you specify.
  103. Drivers are simply a specified group of 'Matchers' which match strings based on specific conditions.
  104. Currently there are only three drivers: Simple, Fuzzy and Levenshtein (Experimental).
  105. ### Simple Search Driver
  106. The Simple search driver only uses 3 matchers each with the relevant multipliers that best suited my testing environments.
  107. ```php
  108. protected $matchers = [
  109. 'TomLingham\Searchy\Matchers\ExactMatcher' => 100,
  110. 'TomLingham\Searchy\Matchers\StartOfStringMatcher' => 50,
  111. 'TomLingham\Searchy\Matchers\InStringMatcher' => 30,
  112. ];
  113. ```
  114. ### Fuzzy Search Driver
  115. The Fuzzy Search Driver is simply another group of matchers setup as follows. The multipliers are what I have used, but feel free to change these or roll your own driver with the same matchers and change the multipliers to suit.
  116. ```php
  117. protected $matchers = [
  118. 'TomLingham\Searchy\Matchers\ExactMatcher' => 100,
  119. 'TomLingham\Searchy\Matchers\StartOfStringMatcher' => 50,
  120. 'TomLingham\Searchy\Matchers\AcronymMatcher' => 42,
  121. 'TomLingham\Searchy\Matchers\ConsecutiveCharactersMatcher' => 40,
  122. 'TomLingham\Searchy\Matchers\StartOfWordsMatcher' => 35,
  123. 'TomLingham\Searchy\Matchers\StudlyCaseMatcher' => 32,
  124. 'TomLingham\Searchy\Matchers\InStringMatcher' => 30,
  125. 'TomLingham\Searchy\Matchers\TimesInStringMatcher' => 8,
  126. ];
  127. ```
  128. ### Levenshtein Search Driver (Experimental)
  129. The Levenshtein Search Driver uses the Levenshetein Distance to calculate the 'distance' between strings. It requires that you have a stored procedure in MySQL similar to the following `levenshtein( string1, string2 )`. There is an SQL file with a suitable function in the `res` folder - feel free to use this one.
  130. ```php
  131. protected $matchers = [
  132. 'TomLingham\Searchy\Matchers\LevenshteinMatcher' => 100
  133. ];
  134. ```
  135. ## Matchers
  136. ### ExactMatcher
  137. Matches an exact string and applies a high multiplier to bring any exact matches to the top.
  138. ### StartOfStringMatcher
  139. Matches Strings that begin with the search string.
  140. For example, a search for 'hel' would match; 'Hello World' or 'helping hand'
  141. ### AcronymMatcher
  142. Matches strings for Acronym 'like' matches but does NOT return Studly Case Matches
  143. For example, a search for 'fb' would match; 'foo bar' or 'Fred Brown' but not 'FreeBeer'.
  144. ### ConsecutiveCharactersMatcher
  145. Matches strings that include all the characters in the search relatively positioned within the string. It also calculates the percentage of characters in the string that are matched and applies the multiplier accordingly.
  146. For Example, a search for 'fba' would match; 'Foo Bar' or 'Afraid of bats', but not 'fabulous'
  147. ### StartOfWordsMatcher
  148. Matches the start of each word against each word in a search.
  149. For example, a search for 'jo ta' would match; 'John Taylor' or 'Joshua B. Takeshi'
  150. ### StudlyCaseMatcher
  151. Matches Studly Case strings using the first letters of the words only
  152. For example a search for 'hp' would match; 'HtmlServiceProvider' or 'HashParser' but not 'hasProvider'
  153. ### InStringMatcher
  154. Matches against any occurrences of a string within a string and is case-insensitive.
  155. For example, a search for 'smi' would match; 'John Smith' or 'Smiley Face'
  156. ### TimesInStringMatcher
  157. Matches a string based on how many times the search string appears inside the string it then applies the multiplier for each occurrence.
  158. For example, a search for 'tha' would match; 'I hope that that cat has caught that mouse' (3 x multiplier) or 'Thanks, it was great!' (1 x multiplier)
  159. ### LevenshteinMatcher
  160. See *Levenshtein Driver*
  161. ## Extending
  162. ### Drivers
  163. It's really easy to roll your own search drivers. Simply create a class that extends `TomLingham\Searchy\SearchDrivers\BaseSearchDriver` and add a property called `$matchers` with an array of matcher classes as the key and the multiplier for each matcher as the values. You can pick from the classes that are already included with Searchy or you can create your own.
  164. ### Matchers
  165. To create your own matchers, you can create your own class that extends `TomLingham\Searchy\Matchers\BaseMatcher` and (for simple Matchers) override the `formatQuery` method to return a string formatted with `%` wildcards in required locations. For more advanced extensions you may need to override the `buildQuery` method and others as well.
  166. ## Contributing & Reporting Bugs
  167. If you would like to improve on the code that is here, feel free to submit a pull request.
  168. If you find any bugs, submit them here and I will respond as soon as possible. Please make sure to include as much information as possible.