Add: get reviews and fix bug
This commit is contained in:
@@ -94,6 +94,7 @@ func NewAPI(config Config) (*API, error) {
|
|||||||
apiMux.HandleFunc("/update_foldername", api.HandleUpdateFoldername)
|
apiMux.HandleFunc("/update_foldername", api.HandleUpdateFoldername)
|
||||||
// review
|
// review
|
||||||
apiMux.HandleFunc("/insert_review", api.HandleInsertReview)
|
apiMux.HandleFunc("/insert_review", api.HandleInsertReview)
|
||||||
|
apiMux.HandleFunc("/get_reviews_on_file", api.HandleGetReviewsOnFile)
|
||||||
// below needs token
|
// below needs token
|
||||||
apiMux.HandleFunc("/walk", api.HandleWalk)
|
apiMux.HandleFunc("/walk", api.HandleWalk)
|
||||||
apiMux.HandleFunc("/reset", api.HandleReset)
|
apiMux.HandleFunc("/reset", api.HandleReset)
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
|
||||||
"msw-open-music/pkg/database"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"msw-open-music/pkg/database"
|
||||||
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -17,13 +17,6 @@ func (api *API) HandleInsertReview(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// check not anonymous
|
|
||||||
err = api.CheckNotAnonymous(w, r)
|
|
||||||
if err != nil {
|
|
||||||
api.HandleError(w, r, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
review.UserId, err = api.GetUserID(w, r)
|
review.UserId, err = api.GetUserID(w, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.HandleError(w, r, err)
|
api.HandleError(w, r, err)
|
||||||
@@ -40,3 +33,37 @@ func (api *API) HandleInsertReview(w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
api.HandleOK(w, r)
|
api.HandleOK(w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GetReviewsOnFileRequest struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GetReviewsOnFileResponse struct {
|
||||||
|
Reviews []*database.Review `json:"reviews"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (api *API) HandleGetReviewsOnFile(w http.ResponseWriter, r *http.Request) {
|
||||||
|
req := &GetReviewsOnFileRequest{}
|
||||||
|
|
||||||
|
err := json.NewDecoder(r.Body).Decode(req)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
reviews, err := api.Db.GetReviewsOnFile(req.ID)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := &GetReviewsOnFileResponse{
|
||||||
|
Reviews: reviews,
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.NewEncoder(w).Encode(resp)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -8,3 +8,35 @@ func (database *Database) InsertReview(review *Review) error {
|
|||||||
review.Content)
|
review.Content)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (database *Database) GetReviewsOnFile(fileId int64) ([]*Review, error) {
|
||||||
|
rows, err := database.stmt.getReviewsOnFile.Query(fileId)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
reviews := make([]*Review, 0)
|
||||||
|
for rows.Next() {
|
||||||
|
review := &Review{
|
||||||
|
User: &User{},
|
||||||
|
File: &File{},
|
||||||
|
}
|
||||||
|
err := rows.Scan(
|
||||||
|
&review.ID,
|
||||||
|
&review.CreatedAt,
|
||||||
|
&review.UpdatedAt,
|
||||||
|
&review.Content,
|
||||||
|
&review.User.ID,
|
||||||
|
&review.User.Username,
|
||||||
|
&review.User.Role,
|
||||||
|
&review.User.AvatarId,
|
||||||
|
&review.File.ID,
|
||||||
|
&review.File.Filename)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
reviews = append(reviews, review)
|
||||||
|
}
|
||||||
|
return reviews, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -207,6 +207,17 @@ var updateFoldernameQuery = `UPDATE folders SET foldername = ? WHERE id = ?;`
|
|||||||
var insertReviewQuery = `INSERT INTO reviews (user_id, file_id, created_at, content)
|
var insertReviewQuery = `INSERT INTO reviews (user_id, file_id, created_at, content)
|
||||||
VALUES (?, ?, ?, ?);`
|
VALUES (?, ?, ?, ?);`
|
||||||
|
|
||||||
|
var getReviewsOnFileQuery = `SELECT
|
||||||
|
reviews.id, reviews.created_at, reviews.updated_at, reviews.content,
|
||||||
|
users.id, users.username, users.role, users.avatar_id,
|
||||||
|
files.id, files.filename
|
||||||
|
FROM reviews
|
||||||
|
JOIN users ON reviews.user_id = users.id
|
||||||
|
JOIN files ON reviews.file_id = files.id
|
||||||
|
WHERE reviews.file_id = ?
|
||||||
|
ORDER BY reviews.created_at
|
||||||
|
;`
|
||||||
|
|
||||||
type Stmt struct {
|
type Stmt struct {
|
||||||
initFilesTable *sql.Stmt
|
initFilesTable *sql.Stmt
|
||||||
initFoldersTable *sql.Stmt
|
initFoldersTable *sql.Stmt
|
||||||
@@ -248,6 +259,7 @@ type Stmt struct {
|
|||||||
deleteTagOnFile *sql.Stmt
|
deleteTagOnFile *sql.Stmt
|
||||||
updateFoldername *sql.Stmt
|
updateFoldername *sql.Stmt
|
||||||
insertReview *sql.Stmt
|
insertReview *sql.Stmt
|
||||||
|
getReviewsOnFile *sql.Stmt
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
||||||
@@ -558,5 +570,11 @@ func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// init getReviewsOnFile
|
||||||
|
stmt.getReviewsOnFile, err = sqlConn.Prepare(getReviewsOnFileQuery)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return stmt, err
|
return stmt, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,6 +35,30 @@ function numberWithCommas(x) {
|
|||||||
return x;
|
return x;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// convert unix timestamp to %Y-%m-%d %H:%M:%S
|
||||||
|
export function convertIntToDateTime(timestamp) {
|
||||||
|
var date = new Date(timestamp * 1000);
|
||||||
|
var year = date.getFullYear();
|
||||||
|
var month = date.getMonth() + 1;
|
||||||
|
var day = date.getDate();
|
||||||
|
var hour = date.getHours();
|
||||||
|
var minute = date.getMinutes();
|
||||||
|
var second = date.getSeconds();
|
||||||
|
var time =
|
||||||
|
year +
|
||||||
|
"-" +
|
||||||
|
(month < 10 ? "0" + month : month) +
|
||||||
|
"-" +
|
||||||
|
(day < 10 ? "0" + day : day) +
|
||||||
|
" " +
|
||||||
|
(hour < 10 ? "0" + hour : hour) +
|
||||||
|
":" +
|
||||||
|
(minute < 10 ? "0" + minute : minute) +
|
||||||
|
":" +
|
||||||
|
(second < 10 ? "0" + second : second);
|
||||||
|
return time;
|
||||||
|
}
|
||||||
|
|
||||||
export function SayHello() {
|
export function SayHello() {
|
||||||
return "Hello";
|
return "Hello";
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,36 @@
|
|||||||
import { useState } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { useParams } from "react-router";
|
import { useParams } from "react-router";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
import { convertIntToDateTime } from "./Common";
|
||||||
|
|
||||||
function ReviewPage() {
|
function ReviewPage() {
|
||||||
let params = useParams();
|
let params = useParams();
|
||||||
const [newReview, setNewReview] = useState("");
|
const [newReview, setNewReview] = useState("");
|
||||||
|
const [reviews, setReviews] = useState([]);
|
||||||
|
|
||||||
|
function refresh() {
|
||||||
|
fetch("/api/v1/get_reviews_on_file", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
id: parseInt(params.id),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
.then((response) => response.json())
|
||||||
|
.then((data) => {
|
||||||
|
if (data.error) {
|
||||||
|
alert(data.error);
|
||||||
|
} else {
|
||||||
|
setReviews(data.reviews);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
refresh();
|
||||||
|
}, []);
|
||||||
|
|
||||||
function submitReview() {
|
function submitReview() {
|
||||||
fetch("/api/v1/insert_review", {
|
fetch("/api/v1/insert_review", {
|
||||||
@@ -22,6 +49,7 @@ function ReviewPage() {
|
|||||||
alert(data.error);
|
alert(data.error);
|
||||||
} else {
|
} else {
|
||||||
setNewReview("");
|
setNewReview("");
|
||||||
|
refresh();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -29,6 +57,17 @@ function ReviewPage() {
|
|||||||
return (
|
return (
|
||||||
<div className="page">
|
<div className="page">
|
||||||
<h3>Review Page</h3>
|
<h3>Review Page</h3>
|
||||||
|
<div>
|
||||||
|
{reviews.map((review) => (
|
||||||
|
<div key={review.id}>
|
||||||
|
<h4>
|
||||||
|
<Link to={`/manage/users/${review.user.id}`}>@{review.user.username}</Link> wrote on{" "}
|
||||||
|
{convertIntToDateTime(review.created_at)}{" "}
|
||||||
|
</h4>
|
||||||
|
<p>{review.content}</p>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<textarea
|
<textarea
|
||||||
value={newReview}
|
value={newReview}
|
||||||
|
|||||||
Reference in New Issue
Block a user