Add basic friend logic
This commit is contained in:
		@@ -31,6 +31,9 @@ func Start(conf common.Conf, db *sql.DB) {
 | 
				
			|||||||
			gv1.GET("/users/self", Method(v1.UserSelfGET, db, common.PrivilegeRead))
 | 
								gv1.GET("/users/self", Method(v1.UserSelfGET, db, common.PrivilegeRead))
 | 
				
			||||||
			gv1.GET("/badges", Method(v1.BadgesGET, db, common.PrivilegeRead))
 | 
								gv1.GET("/badges", Method(v1.BadgesGET, db, common.PrivilegeRead))
 | 
				
			||||||
			gv1.GET("/badges/:id", Method(v1.BadgeByIDGET, db, common.PrivilegeRead))
 | 
								gv1.GET("/badges/:id", Method(v1.BadgeByIDGET, db, common.PrivilegeRead))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								// ReadConfidential privilege required
 | 
				
			||||||
 | 
								gv1.GET("/friends", Method(v1.FriendsGET, db, common.PrivilegeReadConfidential))
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -38,6 +38,7 @@ func BadgesGET(md common.MethodData) (r common.Response) {
 | 
				
			|||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		md.C.Error(err)
 | 
							md.C.Error(err)
 | 
				
			||||||
		r = Err500
 | 
							r = Err500
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	defer rows.Close()
 | 
						defer rows.Close()
 | 
				
			||||||
	for rows.Next() {
 | 
						for rows.Next() {
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										116
									
								
								app/v1/friend.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								app/v1/friend.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,116 @@
 | 
				
			|||||||
 | 
					package v1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"database/sql"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
						"strings"
 | 
				
			||||||
 | 
						"time"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/osuripple/api/common"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type friendData struct {
 | 
				
			||||||
 | 
						userData
 | 
				
			||||||
 | 
						IsMutual bool `json:"is_mutual"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// FriendsGET is the API request handler for GET /friends.
 | 
				
			||||||
 | 
					// It retrieves an user's friends, and whether the friendship is mutual or not.
 | 
				
			||||||
 | 
					func FriendsGET(md common.MethodData) (r common.Response) {
 | 
				
			||||||
 | 
						var myFrienders []int
 | 
				
			||||||
 | 
						myFriendersRaw, err := md.DB.Query("SELECT user1 FROM users_relationships WHERE user2 = ?", md.User.UserID)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							md.C.Error(err)
 | 
				
			||||||
 | 
							r = Err500
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						defer myFriendersRaw.Close()
 | 
				
			||||||
 | 
						for myFriendersRaw.Next() {
 | 
				
			||||||
 | 
							var i int
 | 
				
			||||||
 | 
							err := myFriendersRaw.Scan(&i)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								md.C.Error(err)
 | 
				
			||||||
 | 
								continue
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							myFrienders = append(myFrienders, i)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if err := myFriendersRaw.Err(); err != nil {
 | 
				
			||||||
 | 
							md.C.Error(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Yes.
 | 
				
			||||||
 | 
						myFriendsQuery := `
 | 
				
			||||||
 | 
					SELECT 
 | 
				
			||||||
 | 
						users.id, users.username, users.register_datetime, users.rank, users.latest_activity,
 | 
				
			||||||
 | 
						
 | 
				
			||||||
 | 
						users_stats.username_aka, users_stats.badges_shown,
 | 
				
			||||||
 | 
						users_stats.country, users_stats.show_country
 | 
				
			||||||
 | 
					FROM users_relationships
 | 
				
			||||||
 | 
					LEFT JOIN users
 | 
				
			||||||
 | 
					ON users_relationships.user2 = users.id
 | 
				
			||||||
 | 
					LEFT JOIN users_stats
 | 
				
			||||||
 | 
					ON users_relationships.user2=users_stats.id
 | 
				
			||||||
 | 
					WHERE users_relationships.user1=?
 | 
				
			||||||
 | 
					ORDER BY users_relationships.id`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						results, err := md.DB.Query(myFriendsQuery+common.Paginate(md.C.Query("p"), md.C.Query("l")), md.User.UserID)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							md.C.Error(err)
 | 
				
			||||||
 | 
							r = Err500
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var myFriends []friendData
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						defer results.Close()
 | 
				
			||||||
 | 
						for results.Next() {
 | 
				
			||||||
 | 
							newFriend := friendPuts(md, results)
 | 
				
			||||||
 | 
							for _, uid := range myFrienders {
 | 
				
			||||||
 | 
								if uid == newFriend.ID {
 | 
				
			||||||
 | 
									newFriend.IsMutual = true
 | 
				
			||||||
 | 
									break
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							myFriends = append(myFriends, newFriend)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if err := results.Err(); err != nil {
 | 
				
			||||||
 | 
							md.C.Error(err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func friendPuts(md common.MethodData, row *sql.Rows) (user friendData) {
 | 
				
			||||||
 | 
						var err error
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						registeredOn := int64(0)
 | 
				
			||||||
 | 
						latestActivity := int64(0)
 | 
				
			||||||
 | 
						var badges string
 | 
				
			||||||
 | 
						var showcountry bool
 | 
				
			||||||
 | 
						err = row.Scan(&user.ID, &user.Username, ®isteredOn, &user.Rank, &latestActivity, &user.UsernameAKA, &badges, &user.Country, &showcountry)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							md.C.Error(err)
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						user.RegisteredOn = time.Unix(registeredOn, 0)
 | 
				
			||||||
 | 
						user.LatestActivity = time.Unix(latestActivity, 0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						badgesSl := strings.Split(badges, ",")
 | 
				
			||||||
 | 
						for _, badge := range badgesSl {
 | 
				
			||||||
 | 
							if badge != "" && badge != "0" {
 | 
				
			||||||
 | 
								// We are ignoring errors because who really gives a shit if something's gone wrong on our end in this
 | 
				
			||||||
 | 
								// particular thing, we can just silently ignore this.
 | 
				
			||||||
 | 
								nb, err := strconv.Atoi(badge)
 | 
				
			||||||
 | 
								if err == nil && nb != 0 {
 | 
				
			||||||
 | 
									user.Badges = append(user.Badges, nb)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// If the user wants to stay anonymous, don't show their country.
 | 
				
			||||||
 | 
						// This can be overriden if we have the ReadConfidential privilege and the user we are accessing is the token owner.
 | 
				
			||||||
 | 
						if !(showcountry || (md.User.Privileges.HasPrivilegeReadConfidential() && user.ID == md.User.UserID)) {
 | 
				
			||||||
 | 
							user.Country = "XX"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										33
									
								
								common/paginate.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								common/paginate.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,33 @@
 | 
				
			|||||||
 | 
					package common
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import (
 | 
				
			||||||
 | 
						"fmt"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Paginate creates an additional SQL LIMIT clause for paginating.
 | 
				
			||||||
 | 
					func Paginate(page, limit string) string {
 | 
				
			||||||
 | 
						var (
 | 
				
			||||||
 | 
							pInt int
 | 
				
			||||||
 | 
							lInt int
 | 
				
			||||||
 | 
							err  error
 | 
				
			||||||
 | 
						)
 | 
				
			||||||
 | 
						if page == "" {
 | 
				
			||||||
 | 
							pInt = 1
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							pInt, err = strconv.Atoi(page)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								pInt = 1
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if limit == "" {
 | 
				
			||||||
 | 
							lInt = 50
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							lInt, err = strconv.Atoi(limit)
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								lInt = 50
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						start := (pInt - 1) * lInt
 | 
				
			||||||
 | 
						return fmt.Sprintf(" LIMIT %d,%d ", start, lInt)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
		Reference in New Issue
	
	Block a user