75 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
			
		
		
	
	
			75 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
| <?php
 | |
| 
 | |
| namespace BookStack\Http\Controllers\Api;
 | |
| 
 | |
| use BookStack\Entities\Models\Entity;
 | |
| use BookStack\Search\SearchOptions;
 | |
| use BookStack\Search\SearchResultsFormatter;
 | |
| use BookStack\Search\SearchRunner;
 | |
| use Illuminate\Http\Request;
 | |
| 
 | |
| class SearchApiController extends ApiController
 | |
| {
 | |
|     protected $searchRunner;
 | |
|     protected $resultsFormatter;
 | |
| 
 | |
|     protected $rules = [
 | |
|         'all' => [
 | |
|             'query'  => ['required'],
 | |
|             'page'   => ['integer', 'min:1'],
 | |
|             'count'  => ['integer', 'min:1', 'max:100'],
 | |
|         ],
 | |
|     ];
 | |
| 
 | |
|     public function __construct(SearchRunner $searchRunner, SearchResultsFormatter $resultsFormatter)
 | |
|     {
 | |
|         $this->searchRunner = $searchRunner;
 | |
|         $this->resultsFormatter = $resultsFormatter;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Run a search query against all main content types (shelves, books, chapters & pages)
 | |
|      * in the system. Takes the same input as the main search bar within the BookStack
 | |
|      * interface as a 'query' parameter. See https://www.bookstackapp.com/docs/user/searching/
 | |
|      * for a full list of search term options. Results contain a 'type' property to distinguish
 | |
|      * between: bookshelf, book, chapter & page.
 | |
|      *
 | |
|      * The paging parameters and response format emulates a standard listing endpoint
 | |
|      * but standard sorting and filtering cannot be done on this endpoint. If a count value
 | |
|      * is provided this will only be taken as a suggestion. The results in the response
 | |
|      * may currently be up to 4x this value.
 | |
|      */
 | |
|     public function all(Request $request)
 | |
|     {
 | |
|         $this->validate($request, $this->rules['all']);
 | |
| 
 | |
|         $options = SearchOptions::fromString($request->get('query') ?? '');
 | |
|         $page = intval($request->get('page', '0')) ?: 1;
 | |
|         $count = min(intval($request->get('count', '0')) ?: 20, 100);
 | |
| 
 | |
|         $results = $this->searchRunner->searchEntities($options, 'all', $page, $count);
 | |
|         $this->resultsFormatter->format($results['results']->all(), $options);
 | |
| 
 | |
|         /** @var Entity $result */
 | |
|         foreach ($results['results'] as $result) {
 | |
|             $result->setVisible([
 | |
|                 'id', 'name', 'slug', 'book_id',
 | |
|                 'chapter_id', 'draft', 'template',
 | |
|                 'created_at', 'updated_at',
 | |
|                 'tags', 'type', 'preview_html', 'url',
 | |
|             ]);
 | |
|             $result->setAttribute('type', $result->getType());
 | |
|             $result->setAttribute('url', $result->getUrl());
 | |
|             $result->setAttribute('preview_html', [
 | |
|                 'name'    => (string) $result->getAttribute('preview_name'),
 | |
|                 'content' => (string) $result->getAttribute('preview_content'),
 | |
|             ]);
 | |
|         }
 | |
| 
 | |
|         return response()->json([
 | |
|             'data'  => $results['results'],
 | |
|             'total' => $results['total'],
 | |
|         ]);
 | |
|     }
 | |
| }
 |