2023-03-12 18:00:57 +03:00
// GoToSocial
// Copyright (C) GoToSocial Authors admin@gotosocial.org
// SPDX-License-Identifier: AGPL-3.0-or-later
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
2021-05-29 20:39:43 +03:00
package search
import (
"net/http"
"github.com/gin-gonic/gin"
2023-01-02 15:10:50 +03:00
apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util"
2022-06-08 21:38:03 +03:00
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
2021-05-29 20:39:43 +03:00
"github.com/superseriousbusiness/gotosocial/internal/oauth"
)
2023-07-31 16:47:35 +03:00
// SearchGETHandler swagger:operation GET /api/{api_version}/search searchGet
2021-08-02 20:06:44 +03:00
//
// Search for statuses, accounts, or hashtags, on this instance or elsewhere.
//
// If statuses are in the result, they will be returned in descending chronological order (newest first), with sequential IDs (bigger = newer).
//
2022-09-28 20:30:40 +03:00
// ---
// tags:
// - search
2021-08-02 20:06:44 +03:00
//
2023-06-21 19:26:40 +03:00
// produces:
// - application/json
//
// parameters:
// -
2023-07-31 16:47:35 +03:00
// name: api_version
// type: string
// in: path
// description: >-
// Version of the API to use. Must be either `v1` or `v2`.
// If v1 is used, Hashtag results will be a slice of strings.
// If v2 is used, Hashtag results will be a slice of apimodel tags.
// required: true
// -
2023-06-21 19:26:40 +03:00
// name: max_id
// type: string
// description: >-
// Return only items *OLDER* than the given max ID.
// The item with the specified ID will not be included in the response.
// Currently only used if 'type' is set to a specific type.
// in: query
// required: false
// -
// name: min_id
// type: string
// description: >-
// Return only items *immediately newer* than the given min ID.
// The item with the specified ID will not be included in the response.
// Currently only used if 'type' is set to a specific type.
// in: query
// required: false
// -
// name: limit
// type: integer
// description: Number of each type of item to return.
// default: 20
// maximum: 40
// minimum: 1
// in: query
// required: false
// -
// name: offset
// type: integer
// description: >-
// Page number of results to return (starts at 0).
// This parameter is currently not used, page by selecting
// a specific query type and using maxID and minID instead.
// default: 0
// maximum: 10
// minimum: 0
// in: query
// required: false
// -
// name: q
// type: string
// description: |-
// Query string to search for. This can be in the following forms:
// - `@[username]` -- search for an account with the given username on any domain. Can return multiple results.
// - @[username]@[domain]` -- search for a remote account with exact username and domain. Will only ever return 1 result at most.
// - `https://example.org/some/arbitrary/url` -- search for an account OR a status with the given URL. Will only ever return 1 result at most.
2023-07-31 16:47:35 +03:00
// - `#[hashtag_name]` -- search for a hashtag with the given hashtag name, or starting with the given hashtag name. Case insensitive. Can return multiple results.
2023-06-21 19:26:40 +03:00
// - any arbitrary string -- search for accounts or statuses containing the given string. Can return multiple results.
2024-05-31 13:57:42 +03:00
//
// Arbitrary string queries may include the following operators:
// - `from:localuser`, `from:remoteuser@instance.tld`: restrict results to statuses created by the specified account.
2023-06-21 19:26:40 +03:00
// in: query
// required: true
// -
// name: type
// type: string
// description: |-
// Type of item to return. One of:
// - `` -- empty string; return any/all results.
2023-07-31 16:47:35 +03:00
// - `accounts` -- return only account(s).
// - `statuses` -- return only status(es).
// - `hashtags` -- return only hashtag(s).
2023-06-21 19:26:40 +03:00
// If `type` is specified, paging can be performed using max_id and min_id parameters.
// If `type` is not specified, see the `offset` parameter for paging.
// in: query
// -
// name: resolve
// type: boolean
// description: >-
// If searching query is for `@[username]@[domain]`, or a URL, allow the GoToSocial
// instance to resolve the search by making calls to remote instances (webfinger, ActivityPub, etc).
// default: false
// in: query
// -
// name: following
// type: boolean
// description: >-
// If search type includes accounts, and search query is an arbitrary string, show only accounts
// that the requesting account follows. If this is set to `true`, then the GoToSocial instance will
// enhance the search by also searching within account notes, not just in usernames and display names.
// default: false
// in: query
// -
// name: exclude_unreviewed
// type: boolean
// description: >-
// If searching for hashtags, exclude those not yet approved by instance admin.
// Currently this parameter is unused.
// default: false
// in: query
2024-05-31 13:57:42 +03:00
// -
// name: account_id
// type: string
// description: >-
// Restrict results to statuses created by the specified account.
// in: query
2023-06-21 19:26:40 +03:00
//
2022-09-28 20:30:40 +03:00
// security:
// - OAuth2 Bearer:
// - read:search
2021-08-02 20:06:44 +03:00
//
2022-09-28 20:30:40 +03:00
// responses:
// '200':
// name: search results
// description: Results of the search.
// schema:
2023-07-31 16:47:35 +03:00
// "$ref": "#/definitions/searchResult"
2022-09-28 20:30:40 +03:00
// '400':
// description: bad request
// '401':
// description: unauthorized
// '404':
// description: not found
// '406':
// description: not acceptable
// '500':
// description: internal server error
2021-05-29 20:39:43 +03:00
func ( m * Module ) SearchGETHandler ( c * gin . Context ) {
2023-07-31 16:47:35 +03:00
apiVersion , errWithCode := apiutil . ParseAPIVersion (
c . Param ( apiutil . APIVersionKey ) ,
[ ] string { apiutil . APIv1 , apiutil . APIv2 } ... ,
)
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
}
2022-06-08 21:38:03 +03:00
authed , err := oauth . Authed ( c , true , true , true , true )
2021-05-29 20:39:43 +03:00
if err != nil {
2023-02-02 16:08:13 +03:00
apiutil . ErrorHandler ( c , gtserror . NewErrorUnauthorized ( err , err . Error ( ) ) , m . processor . InstanceGetV1 )
2021-05-29 20:39:43 +03:00
return
}
2024-03-13 15:53:29 +03:00
if authed . Account . IsMoving ( ) {
// For moving/moved accounts, just return
// empty to avoid breaking client apps.
results := & apimodel . SearchResult {
Accounts : make ( [ ] * apimodel . Account , 0 ) ,
Statuses : make ( [ ] * apimodel . Status , 0 ) ,
Hashtags : make ( [ ] any , 0 ) ,
}
apiutil . JSON ( c , http . StatusOK , results )
return
}
2023-01-02 15:10:50 +03:00
if _ , err := apiutil . NegotiateAccept ( c , apiutil . JSONAcceptHeaders ... ) ; err != nil {
2023-02-02 16:08:13 +03:00
apiutil . ErrorHandler ( c , gtserror . NewErrorNotAcceptable ( err , err . Error ( ) ) , m . processor . InstanceGetV1 )
2021-12-11 19:50:00 +03:00
return
}
2023-06-21 19:26:40 +03:00
limit , errWithCode := apiutil . ParseLimit ( c . Query ( apiutil . LimitKey ) , 20 , 40 , 1 )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
offset , errWithCode := apiutil . ParseSearchOffset ( c . Query ( apiutil . SearchOffsetKey ) , 0 , 10 , 0 )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
2021-05-29 20:39:43 +03:00
return
}
2023-06-21 19:26:40 +03:00
query , errWithCode := apiutil . ParseSearchQuery ( c . Query ( apiutil . SearchQueryKey ) )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
resolve , errWithCode := apiutil . ParseSearchResolve ( c . Query ( apiutil . SearchResolveKey ) , false )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
following , errWithCode := apiutil . ParseSearchFollowing ( c . Query ( apiutil . SearchFollowingKey ) , false )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
excludeUnreviewed , errWithCode := apiutil . ParseSearchExcludeUnreviewed ( c . Query ( apiutil . SearchExcludeUnreviewedKey ) , false )
if errWithCode != nil {
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
return
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
searchRequest := & apimodel . SearchRequest {
MaxID : c . Query ( apiutil . MaxIDKey ) ,
MinID : c . Query ( apiutil . MinIDKey ) ,
2021-05-29 20:39:43 +03:00
Limit : limit ,
Offset : offset ,
2023-06-21 19:26:40 +03:00
Query : query ,
QueryType : c . Query ( apiutil . SearchTypeKey ) ,
Resolve : resolve ,
2021-05-29 20:39:43 +03:00
Following : following ,
2023-06-21 19:26:40 +03:00
ExcludeUnreviewed : excludeUnreviewed ,
2024-05-31 13:57:42 +03:00
AccountID : c . Query ( apiutil . SearchAccountIDKey ) ,
2023-07-31 16:47:35 +03:00
APIv1 : apiVersion == apiutil . APIv1 ,
2021-05-29 20:39:43 +03:00
}
2023-06-21 19:26:40 +03:00
results , errWithCode := m . processor . Search ( ) . Get ( c . Request . Context ( ) , authed . Account , searchRequest )
2021-05-29 20:39:43 +03:00
if errWithCode != nil {
2023-02-02 16:08:13 +03:00
apiutil . ErrorHandler ( c , errWithCode , m . processor . InstanceGetV1 )
2021-05-29 20:39:43 +03:00
return
}
2023-11-27 17:00:57 +03:00
apiutil . JSON ( c , http . StatusOK , results )
2021-05-29 20:39:43 +03:00
}