search.go 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. package route
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "strings"
  8. "github.com/G-Node/gogs/internal/context"
  9. "github.com/G-Node/gogs/internal/db"
  10. "github.com/G-Node/gogs/internal/setting"
  11. "github.com/G-Node/libgin/libgin"
  12. log "gopkg.in/clog.v1"
  13. )
  14. const (
  15. EXPLORE_DATA = "explore/data"
  16. EXPLORE_COMMITS = "explore/commits"
  17. )
  18. type set map[int64]interface{}
  19. func newset() set {
  20. return make(map[int64]interface{}, 0)
  21. }
  22. func (s set) add(item int64) {
  23. s[item] = nil
  24. }
  25. func (s set) contains(item int64) bool {
  26. _, yes := s[item]
  27. return yes
  28. }
  29. func (s set) remove(item int64) {
  30. delete(s, item)
  31. }
  32. func (s set) asSlice() []int64 {
  33. slice := make([]int64, len(s))
  34. idx := 0
  35. for item := range s {
  36. slice[idx] = item
  37. idx++
  38. }
  39. return slice
  40. }
  41. func collectSearchableRepoIDs(c *context.Context) ([]int64, error) {
  42. repoIDSet := newset()
  43. updateSet := func(repos []*db.Repository) {
  44. for _, r := range repos {
  45. repoIDSet.add(r.ID)
  46. }
  47. }
  48. if c.User != nil {
  49. ownRepos := c.User.Repos // user's own repositories
  50. updateSet(ownRepos)
  51. accessibleRepos, _ := c.User.GetAccessibleRepositories(0) // shared and org repos
  52. updateSet(accessibleRepos)
  53. }
  54. // Run a full repository search (with no keywords) to get public
  55. // repositories and then filter out the unlisted ones.
  56. repos, _, err := db.SearchRepositoryByName(&db.SearchRepoOptions{
  57. Keyword: "",
  58. UserID: c.UserID(),
  59. OrderBy: "updated_unix DESC",
  60. Page: 0,
  61. PageSize: 0,
  62. })
  63. if err != nil {
  64. c.ServerError("SearchRepositoryByName", err)
  65. return nil, err
  66. }
  67. // If it's not unlisted, add it to the set
  68. // This will add public (listed) repositories
  69. for _, r := range repos {
  70. if !r.Unlisted {
  71. repoIDSet.add(r.ID)
  72. }
  73. }
  74. return repoIDSet.asSlice(), nil
  75. }
  76. func search(c *context.Context, keywords string, sType int) ([]byte, error) {
  77. if setting.Search.SearchURL == "" {
  78. log.Error(2, "Unable to perform search: SearchURL not configured")
  79. return nil, fmt.Errorf("Extended search not implemented")
  80. }
  81. key := []byte(setting.Search.Key)
  82. repoids, err := collectSearchableRepoIDs(c)
  83. if err != nil {
  84. log.Error(2, "Failed to collect searchable repository IDs: %v", err)
  85. return nil, err
  86. }
  87. searchdata := libgin.SearchRequest{Keywords: keywords, SType: sType, RepoIDs: repoids}
  88. data, err := json.Marshal(searchdata)
  89. if err != nil {
  90. log.Error(2, "Failed to marshal search request for gin-dex: %v", err)
  91. return nil, err
  92. }
  93. // encrypt query
  94. encdata, err := libgin.EncryptString(key, string(data))
  95. if err != nil {
  96. log.Error(2, "Failed to encrypt search data for gin-dex: %v", err)
  97. return nil, err
  98. }
  99. // Send query to gin-dex
  100. req, err := http.NewRequest("POST", setting.Search.SearchURL, strings.NewReader(encdata))
  101. if err != nil {
  102. log.Error(2, "Failed to build request for gin-dex: %v", err)
  103. }
  104. cl := http.Client{}
  105. resp, err := cl.Do(req)
  106. if err != nil {
  107. log.Error(2, "Failed to send request to gin-dex: %v", err)
  108. return nil, err
  109. }
  110. encrespdata, err := ioutil.ReadAll(resp.Body) // response is encrypted
  111. if err != nil {
  112. log.Error(2, "Failed to read response body from gin-dex: %v", err)
  113. return nil, err
  114. }
  115. // decrypt response
  116. respdata, err := libgin.DecryptString(key, string(encrespdata))
  117. if err != nil {
  118. log.Error(2, "Failed to decrypt response body form gin-dex: %v", err)
  119. return nil, err
  120. }
  121. return []byte(respdata), nil
  122. }
  123. // ExploreData handles the search box served at /explore/data
  124. func ExploreData(c *context.Context) {
  125. keywords := c.Query("q")
  126. sType := c.QueryInt("stype") // non integer stype will return 0
  127. c.Data["Title"] = c.Tr("explore")
  128. c.Data["PageIsExplore"] = true
  129. c.Data["PageIsExploreData"] = true
  130. // send query data back even if the search fails or is aborted to fill in
  131. // the form on refresh
  132. c.Data["Keywords"] = keywords
  133. c.Data["opsel"] = sType
  134. res := libgin.SearchResults{}
  135. if keywords == "" {
  136. // no keywords submitted: don't search
  137. log.Trace("Loading empty data search page")
  138. c.Data["Blobs"] = res.Blobs
  139. c.HTML(200, EXPLORE_DATA)
  140. return
  141. }
  142. log.Trace("Searching data/blobs")
  143. data, err := search(c, keywords, sType)
  144. if err != nil {
  145. log.Error(2, "Query returned error: %v", err)
  146. c.Data["Blobs"] = res.Blobs
  147. c.HTML(200, EXPLORE_DATA)
  148. return
  149. }
  150. err = json.Unmarshal(data, &res)
  151. if err != nil {
  152. log.Error(2, "Failed to unmarshal response: %v", err)
  153. c.Data["Blobs"] = res.Blobs
  154. c.HTML(200, EXPLORE_DATA)
  155. return
  156. }
  157. c.Data["Blobs"] = res.Blobs
  158. c.HTML(200, EXPLORE_DATA)
  159. }
  160. // ExploreCommits handles the search box served at /explore/commits
  161. func ExploreCommits(c *context.Context) {
  162. keywords := c.Query("q")
  163. sType := c.QueryInt("stype") // non integer stype will return 0
  164. c.Data["Title"] = c.Tr("explore")
  165. c.Data["PageIsExplore"] = true
  166. c.Data["PageIsExploreCommits"] = true
  167. // send query data back even if the search fails or is aborted to fill in
  168. // the form on refresh
  169. c.Data["Keywords"] = keywords
  170. c.Data["opsel"] = sType
  171. res := libgin.SearchResults{}
  172. if keywords == "" {
  173. log.Trace("Loading empty commit search page")
  174. // no keywords submitted: don't search
  175. c.Data["Commits"] = res.Commits
  176. c.HTML(200, EXPLORE_COMMITS)
  177. return
  178. }
  179. log.Trace("Searching commits")
  180. data, err := search(c, keywords, sType)
  181. if err != nil {
  182. log.Error(2, "Query returned error: %v", err)
  183. c.Data["Commits"] = res.Commits
  184. c.HTML(200, EXPLORE_COMMITS)
  185. }
  186. err = json.Unmarshal(data, &res)
  187. if err != nil {
  188. log.Error(2, "Failed to unmarshal response: %v", err)
  189. c.Data["Commits"] = res.Commits
  190. c.HTML(200, EXPLORE_COMMITS)
  191. return
  192. }
  193. c.Data["Commits"] = res.Commits
  194. c.HTML(200, EXPLORE_COMMITS)
  195. }