b8be14ea8d
* Add a world cities database with type-ahead search on the Member Directory. * Users can search for a known city to order users by distance from that city rather than from their own configured location on their settings page. * Users must opt-in their own location before this feature may be used, in order to increase adoption of the location feature and to enforce fairness. * The `nonshy setup locations` command can import the world cities database.
241 lines
7.0 KiB
Go
241 lines
7.0 KiB
Go
package account
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"code.nonshy.com/nonshy/website/pkg/config"
|
|
"code.nonshy.com/nonshy/website/pkg/geoip"
|
|
"code.nonshy.com/nonshy/website/pkg/log"
|
|
"code.nonshy.com/nonshy/website/pkg/models"
|
|
"code.nonshy.com/nonshy/website/pkg/session"
|
|
"code.nonshy.com/nonshy/website/pkg/spam"
|
|
"code.nonshy.com/nonshy/website/pkg/templates"
|
|
"code.nonshy.com/nonshy/website/pkg/worker"
|
|
)
|
|
|
|
// Search controller.
|
|
func Search() http.HandlerFunc {
|
|
tmpl := templates.Must("account/search.html")
|
|
|
|
// Whitelist for ordering options.
|
|
var sortWhitelist = []string{
|
|
"last_login_at desc",
|
|
"created_at desc",
|
|
"username",
|
|
"username desc",
|
|
"lower(name)",
|
|
"lower(name) desc",
|
|
"distance",
|
|
}
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
// Search filters.
|
|
var (
|
|
isCertified = r.FormValue("certified")
|
|
username = r.FormValue("name") // username search
|
|
searchTerm = r.FormValue("search") // profile text search
|
|
citySearch = r.FormValue("wcs")
|
|
gender = r.FormValue("gender")
|
|
orientation = r.FormValue("orientation")
|
|
maritalStatus = r.FormValue("marital_status")
|
|
hereFor = r.FormValue("here_for")
|
|
friendSearch = r.FormValue("friends") == "true"
|
|
likedSearch = r.FormValue("liked") == "true"
|
|
sort = r.FormValue("sort")
|
|
sortOK bool
|
|
)
|
|
|
|
ageMin, err1 := strconv.Atoi(r.FormValue("age_min"))
|
|
ageMax, err2 := strconv.Atoi(r.FormValue("age_max"))
|
|
if ageMin > ageMax && err1 == nil && err2 == nil {
|
|
ageMin, ageMax = ageMax, ageMin
|
|
}
|
|
|
|
rawSearch := models.ParseSearchString(searchTerm)
|
|
search, restricted := spam.RestrictSearchTerms(rawSearch)
|
|
|
|
// Get current user.
|
|
currentUser, err := session.CurrentUser(r)
|
|
if err != nil {
|
|
session.FlashError(w, r, "Couldn't get current user!")
|
|
templates.Redirect(w, "/")
|
|
return
|
|
}
|
|
|
|
// Report when search terms are restricted.
|
|
if restricted != nil {
|
|
// Admin users: allow the search anyway.
|
|
if currentUser.IsAdmin {
|
|
search = rawSearch
|
|
} else {
|
|
fb := &models.Feedback{
|
|
Intent: "report",
|
|
Subject: "Search Keyword Blacklist",
|
|
UserID: currentUser.ID,
|
|
TableName: "users",
|
|
TableID: currentUser.ID,
|
|
Message: fmt.Sprintf(
|
|
"A user has run a search on the Member Directory using search terms which are prohibited.\n\n"+
|
|
"Their search query was: %s",
|
|
searchTerm,
|
|
),
|
|
}
|
|
|
|
// Save the feedback.
|
|
if err := models.CreateFeedback(fb); err != nil {
|
|
log.Error("Couldn't save feedback from user updating their DOB: %s", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Geolocation/Who's Nearby: if the current user uses GeoIP, update
|
|
// their coordinates now.
|
|
myLocation, err := models.RefreshGeoIP(currentUser.ID, r)
|
|
if err != nil {
|
|
log.Error("RefreshGeoIP: %s", err)
|
|
}
|
|
|
|
// Are they doing a Location search (from world city typeahead)?
|
|
var city *models.WorldCities
|
|
if citySearch != "" {
|
|
sort = "distance"
|
|
|
|
// Require the current user to have THEIR location set, for fairness.
|
|
if myLocation.Source == models.LocationSourceNone {
|
|
session.FlashError(w, r, "You must set your own location before you can search for others by their location.")
|
|
} else {
|
|
// Look up the coordinates of their search.
|
|
city, err = models.FindWorldCity(citySearch)
|
|
if err != nil {
|
|
session.FlashError(w, r, "Location search: no match was found for '%s', please use one of the exact search results from the type-ahead on the Location field.", citySearch)
|
|
citySearch = "" // null out their search
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sort options.
|
|
for _, v := range sortWhitelist {
|
|
if sort == v {
|
|
sortOK = true
|
|
break
|
|
}
|
|
}
|
|
if !sortOK {
|
|
sort = "last_login_at desc"
|
|
}
|
|
|
|
// Default
|
|
if isCertified == "" {
|
|
isCertified = "true"
|
|
}
|
|
|
|
// Always filter for certified-only users unless the request specifically looked for non-certified.
|
|
// Searches for disabled/banned users (admin only) should also reveal ALL users including non-certified.
|
|
var certifiedOnly = true
|
|
if isCertified == "false" || isCertified == "all" || isCertified == "disabled" || isCertified == "banned" {
|
|
certifiedOnly = false
|
|
}
|
|
|
|
// Non-admin view: always hide non-certified profiles, they can be unsafe (fake profiles, scams if they won't certify)
|
|
if !currentUser.IsAdmin {
|
|
certifiedOnly = true
|
|
}
|
|
|
|
pager := &models.Pagination{
|
|
PerPage: config.PageSizeMemberSearch,
|
|
Sort: sort,
|
|
}
|
|
pager.ParsePage(r)
|
|
|
|
users, err := models.SearchUsers(currentUser, &models.UserSearch{
|
|
Username: username,
|
|
Gender: gender,
|
|
Orientation: orientation,
|
|
MaritalStatus: maritalStatus,
|
|
HereFor: hereFor,
|
|
ProfileText: search,
|
|
Certified: certifiedOnly,
|
|
NearCity: city,
|
|
NotCertified: isCertified == "false",
|
|
InnerCircle: isCertified == "circle",
|
|
ShyAccounts: isCertified == "shy",
|
|
IsBanned: isCertified == "banned",
|
|
IsDisabled: isCertified == "disabled",
|
|
IsAdmin: isCertified == "admin",
|
|
Friends: friendSearch,
|
|
Liked: likedSearch,
|
|
AgeMin: ageMin,
|
|
AgeMax: ageMax,
|
|
}, pager)
|
|
if err != nil {
|
|
session.FlashError(w, r, "An error has occurred: %s.", err)
|
|
}
|
|
|
|
// Who's Nearby feature, get some data.
|
|
insights, _ := geoip.GetRequestInsights(r)
|
|
|
|
// Collect usernames to map to chat online status.
|
|
var usernames = []string{}
|
|
var userIDs = []uint64{}
|
|
for _, user := range users {
|
|
usernames = append(usernames, user.Username)
|
|
userIDs = append(userIDs, user.ID)
|
|
}
|
|
|
|
// User IDs of these I have "Liked"
|
|
likedIDs, err := models.LikedIDs(currentUser, "users", userIDs)
|
|
if err != nil {
|
|
log.Error("LikedIDs: %s", err)
|
|
}
|
|
|
|
var vars = map[string]interface{}{
|
|
"Users": users,
|
|
"Pager": pager,
|
|
"Enum": config.ProfileEnums,
|
|
|
|
// Search filter values.
|
|
"Certified": isCertified,
|
|
"Gender": gender,
|
|
"Orientation": orientation,
|
|
"MaritalStatus": maritalStatus,
|
|
"HereFor": hereFor,
|
|
"EmailOrUsername": username,
|
|
"Search": searchTerm,
|
|
"City": citySearch,
|
|
"AgeMin": ageMin,
|
|
"AgeMax": ageMax,
|
|
"FriendSearch": friendSearch,
|
|
"LikedSearch": likedSearch,
|
|
"Sort": sort,
|
|
|
|
// Restricted Search errors.
|
|
"RestrictedSearchError": restricted,
|
|
|
|
// Photo counts mapped to users
|
|
"PhotoCountMap": models.MapPhotoCounts(users),
|
|
|
|
// Map Shy Account badges for these results
|
|
"ShyMap": models.MapShyAccounts(users),
|
|
|
|
// Map friendships and likes to these users.
|
|
"FriendMap": models.MapFriends(currentUser, users),
|
|
"LikedMap": models.MapLikes(currentUser, "users", likedIDs),
|
|
|
|
// Users on the chat room map.
|
|
"UserOnChatMap": worker.GetChatStatistics().MapUsersOnline(usernames),
|
|
|
|
// Current user's location setting.
|
|
"MyLocation": myLocation,
|
|
"GeoIPInsights": insights,
|
|
"DistanceMap": models.MapDistances(currentUser, city, users),
|
|
}
|
|
|
|
if err := tmpl.Execute(w, r, vars); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
})
|
|
}
|