QA: Add new script to migrate QA content from a DB to another
This commit is contained in:
parent
2cf9723c6c
commit
f4dcaa23a3
84
admin/api/qa.go
Normal file
84
admin/api/qa.go
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"srs.epita.fr/fic-server/libfic"
|
||||||
|
|
||||||
|
"github.com/julienschmidt/httprouter"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
router.POST("/api/qa/", apiHandler(importExerciceQA))
|
||||||
|
router.POST("/api/qa/:qid/comments", apiHandler(qaHandler(importQAComment)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func qaHandler(f func(fic.QAQuery, []byte) (interface{}, error)) func(httprouter.Params, []byte) (interface{}, error) {
|
||||||
|
return func(ps httprouter.Params, body []byte) (interface{}, error) {
|
||||||
|
if qid, err := strconv.ParseInt(string(ps.ByName("qid")), 10, 64); err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else if query, err := fic.GetQAQuery(qid); err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else {
|
||||||
|
return f(query, body)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func importExerciceQA(_ httprouter.Params, body []byte) (interface{}, error) {
|
||||||
|
// Create a new query
|
||||||
|
var uq fic.QAQuery
|
||||||
|
if err := json.Unmarshal(body, &uq); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var exercice fic.Exercice
|
||||||
|
var err error
|
||||||
|
if uq.IdExercice == 0 {
|
||||||
|
return nil, errors.New("id_exercice not filled")
|
||||||
|
} else if exercice, err = fic.GetExercice(uq.IdExercice); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(uq.State) == 0 {
|
||||||
|
return nil, errors.New("State not filled")
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(uq.Subject) == 0 {
|
||||||
|
return nil, errors.New("Subject not filled")
|
||||||
|
}
|
||||||
|
|
||||||
|
if qa, err := exercice.NewQAQuery(uq.Subject, uq.IdTeam, uq.User, uq.State); err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else {
|
||||||
|
qa.Creation = uq.Creation
|
||||||
|
qa.Solved = uq.Solved
|
||||||
|
qa.Closed = qa.Closed
|
||||||
|
|
||||||
|
_, err = qa.Update()
|
||||||
|
return qa, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func importQAComment(query fic.QAQuery, body []byte) (interface{}, error) {
|
||||||
|
// Create a new query
|
||||||
|
var uc fic.QAComment
|
||||||
|
if err := json.Unmarshal(body, &uc); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(uc.Content) == 0 {
|
||||||
|
return nil, errors.New("Empty comment")
|
||||||
|
}
|
||||||
|
|
||||||
|
if qac, err := query.AddComment(uc.Content, uc.IdTeam, uc.User); err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else {
|
||||||
|
qac.Date = uc.Date
|
||||||
|
|
||||||
|
_, err = qac.Update()
|
||||||
|
return qac, err
|
||||||
|
}
|
||||||
|
}
|
@ -422,7 +422,7 @@ CREATE TABLE IF NOT EXISTS claim_descriptions(
|
|||||||
CREATE TABLE IF NOT EXISTS exercices_qa(
|
CREATE TABLE IF NOT EXISTS exercices_qa(
|
||||||
id_qa INTEGER NOT NULL PRIMARY KEY AUTO_INCREMENT,
|
id_qa INTEGER NOT NULL PRIMARY KEY AUTO_INCREMENT,
|
||||||
id_exercice INTEGER NOT NULL,
|
id_exercice INTEGER NOT NULL,
|
||||||
id_team INTEGER NOT NULL,
|
id_team INTEGER NULL,
|
||||||
authuser VARCHAR(255) NOT NULL,
|
authuser VARCHAR(255) NOT NULL,
|
||||||
subject VARCHAR(255) NOT NULL,
|
subject VARCHAR(255) NOT NULL,
|
||||||
creation TIMESTAMP NOT NULL,
|
creation TIMESTAMP NOT NULL,
|
||||||
@ -439,7 +439,7 @@ CREATE TABLE IF NOT EXISTS exercices_qa(
|
|||||||
CREATE TABLE IF NOT EXISTS qa_comments(
|
CREATE TABLE IF NOT EXISTS qa_comments(
|
||||||
id_comment INTEGER NOT NULL PRIMARY KEY AUTO_INCREMENT,
|
id_comment INTEGER NOT NULL PRIMARY KEY AUTO_INCREMENT,
|
||||||
id_qa INTEGER NOT NULL,
|
id_qa INTEGER NOT NULL,
|
||||||
id_team INTEGER NOT NULL,
|
id_team INTEGER NULL,
|
||||||
authuser VARCHAR(255) NOT NULL,
|
authuser VARCHAR(255) NOT NULL,
|
||||||
date TIMESTAMP NOT NULL,
|
date TIMESTAMP NOT NULL,
|
||||||
content TEXT NOT NULL,
|
content TEXT NOT NULL,
|
||||||
|
@ -9,7 +9,7 @@ import (
|
|||||||
type QAQuery struct {
|
type QAQuery struct {
|
||||||
Id int64 `json:"id"`
|
Id int64 `json:"id"`
|
||||||
IdExercice int64 `json:"id_exercice"`
|
IdExercice int64 `json:"id_exercice"`
|
||||||
IdTeam int64 `json:"id_team"`
|
IdTeam *int64 `json:"id_team"`
|
||||||
User string `json:"user"`
|
User string `json:"user"`
|
||||||
Creation time.Time `json:"creation"`
|
Creation time.Time `json:"creation"`
|
||||||
State string `json:"state"`
|
State string `json:"state"`
|
||||||
@ -91,7 +91,7 @@ func (e Exercice) GetQAQuery(id int64) (q QAQuery, err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewQAQuery creates and fills a new struct QAQuery and registers it into the database.
|
// NewQAQuery creates and fills a new struct QAQuery and registers it into the database.
|
||||||
func (e Exercice) NewQAQuery(subject string, teamId int64, user string, state string) (QAQuery, error) {
|
func (e Exercice) NewQAQuery(subject string, teamId *int64, user string, state string) (QAQuery, error) {
|
||||||
if res, err := DBExec("INSERT INTO exercices_qa (id_exercice, id_team, authuser, creation, state, subject) VALUES (?, ?, ?, ?, ?, ?)", e.Id, teamId, user, time.Now(), state, subject); err != nil {
|
if res, err := DBExec("INSERT INTO exercices_qa (id_exercice, id_team, authuser, creation, state, subject) VALUES (?, ?, ?, ?, ?, ?)", e.Id, teamId, user, time.Now(), state, subject); err != nil {
|
||||||
return QAQuery{}, err
|
return QAQuery{}, err
|
||||||
} else if qid, err := res.LastInsertId(); err != nil {
|
} else if qid, err := res.LastInsertId(); err != nil {
|
||||||
@ -139,7 +139,7 @@ func ClearQAQueries() (int64, error) {
|
|||||||
// QAComment represents some text describing a QAQuery.
|
// QAComment represents some text describing a QAQuery.
|
||||||
type QAComment struct {
|
type QAComment struct {
|
||||||
Id int64 `json:"id"`
|
Id int64 `json:"id"`
|
||||||
IdTeam int64 `json:"id_team"`
|
IdTeam *int64 `json:"id_team"`
|
||||||
User string `json:"user"`
|
User string `json:"user"`
|
||||||
Date time.Time `json:"date"`
|
Date time.Time `json:"date"`
|
||||||
Content string `json:"content"`
|
Content string `json:"content"`
|
||||||
@ -172,7 +172,7 @@ func (q QAQuery) GetComment(id int64) (c QAComment, err error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// AddComment append in the database a new description; then returns the corresponding structure.
|
// AddComment append in the database a new description; then returns the corresponding structure.
|
||||||
func (q QAQuery) AddComment(content string, teamId int64, user string) (QAComment, error) {
|
func (q QAQuery) AddComment(content string, teamId *int64, user string) (QAComment, error) {
|
||||||
if res, err := DBExec("INSERT INTO qa_comments (id_qa, id_team, authuser, date, content) VALUES (?, ?, ?, ?, ?)", q.Id, teamId, user, time.Now(), content); err != nil {
|
if res, err := DBExec("INSERT INTO qa_comments (id_qa, id_team, authuser, date, content) VALUES (?, ?, ?, ?, ?)", q.Id, teamId, user, time.Now(), content); err != nil {
|
||||||
return QAComment{}, err
|
return QAComment{}, err
|
||||||
} else if cid, err := res.LastInsertId(); err != nil {
|
} else if cid, err := res.LastInsertId(); err != nil {
|
||||||
|
27
qa-fill-todo.sh
Executable file
27
qa-fill-todo.sh
Executable file
@ -0,0 +1,27 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
NB_THEMES_TODO=3
|
||||||
|
MIN_LVL_TODO=1
|
||||||
|
MAX_LVL_TODO=3
|
||||||
|
|
||||||
|
|
||||||
|
MIN_EXO_TODO=$(($MIN_LVL_TODO - 1))
|
||||||
|
MAX_EXO_TODO=$(($MAX_LVL_TODO - 1))
|
||||||
|
|
||||||
|
curl -s http://127.0.0.1:8081/api/teams/ | jq -r '.[] | [(.id | tostring), " ", .name] | add' | while read TEAMID TEAMSTR
|
||||||
|
do
|
||||||
|
if echo $TEAMSTR | grep "FIC Groupe" > /dev/null 2> /dev/null
|
||||||
|
then
|
||||||
|
GRPFIC=$(echo $TEAMSTR | sed -r 's/FIC Groupe //')
|
||||||
|
|
||||||
|
THEMES_TO_TESTS=$(curl -s http://127.0.0.1:8081/api/themes | jq -r '.[] | [(.id | tostring), " ", (.path | split("-") | .[0])] | add' | grep -v " $GRPFIC" | shuf | head -n $NB_THEMES_TODO | cut -d " " -f 1 | xargs | sed 's/^/"/;s/ /","/g;s/$/"/')
|
||||||
|
|
||||||
|
curl -s http://127.0.0.1:8081/api/themes.json | jq -r '.['$THEMES_TO_TESTS'] | .exercices | keys | .['$MIN_EXO_TODO:$MAX_EXO_TODO'] | .[]' | while read EXID
|
||||||
|
do
|
||||||
|
#curl -X POST -d @- -H "X-FIC-Team: nemunaire" http://127.0.0.1:8083/api/qa_work.json <<EOF
|
||||||
|
cat <<EOF
|
||||||
|
{"id_team": $TEAMID, "id_exercice": $EXID}
|
||||||
|
EOF
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
done
|
16
qa-fill-view.sh
Executable file
16
qa-fill-view.sh
Executable file
@ -0,0 +1,16 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Register Teams association betweeen "FIC Groupe X" and X, the theme directory prefix
|
||||||
|
`curl -s http://127.0.0.1:8081/api/teams/ | jq -r '.[] | [(.id | tostring), " ", .name] | add' | while read ID TEAM; do if echo $TEAM | grep "FIC Groupe" > /dev/null 2> /dev/null; then echo TEAM_$TEAM=$ID | sed -r 's/FIC Groupe //'; fi; done`
|
||||||
|
|
||||||
|
# Add their themes and exercices
|
||||||
|
curl -s http://127.0.0.1:8081/api/themes | jq '.[].id' | while read tid
|
||||||
|
do
|
||||||
|
TEAM="TEAM_$(curl -s http://127.0.0.1:8081/api/themes/$tid | jq -r .path | sed -r 's/-.*$//')"
|
||||||
|
curl -s http://127.0.0.1:8081/api/themes/$tid/exercices | jq .[].id | while read ex
|
||||||
|
do
|
||||||
|
curl -X POST -d @- -H "X-FIC-Team: nemunaire" http://127.0.0.1:8083/api/qa_my_exercices.json <<EOF
|
||||||
|
{"id_team": ${!TEAM}, "id_exercice": $ex}
|
||||||
|
EOF
|
||||||
|
done
|
||||||
|
done
|
70
qa-merge-db.sh
Executable file
70
qa-merge-db.sh
Executable file
@ -0,0 +1,70 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
[ $# -lt 1 ] && { >&2 echo "Please give DB to read from as argument"; exit 1; }
|
||||||
|
|
||||||
|
MYSQL_USER="${MYSQL_USER:-fic}"
|
||||||
|
MYSQL_PASSWORD="${MYSQL_PASSWORD:-fic}"
|
||||||
|
MYSQL_DATABASE="$1"
|
||||||
|
|
||||||
|
QA_ADMIN="${QA_ADMIN:-nemunaire@nemunai.re}"
|
||||||
|
|
||||||
|
EXERCICES=$(curl -s -H "X-FIC-Team: ${QA_ADMIN}" http://127.0.0.1:8083/api/exercices/)
|
||||||
|
TEAMS=$(curl -s http://127.0.0.1:8081/api/teams/ | jq -r '.[].id' | while read IDTEAM; do curl -s http://127.0.0.1:8081/api/teams/$IDTEAM/associations | jq -r ".[] | {login: ., id_team: \"$IDTEAM\"}"; done)
|
||||||
|
|
||||||
|
echo "select Q.*, T.name, E.title from exercices_qa Q INNER JOIN exercices E ON E.id_exercice = Q.id_exercice INNER JOIN themes T ON T.id_theme = E.id_theme;" | mysql --skip-column-names -u "${MYSQL_USER}" --password="${MYSQL_PASSWORD}" "${MYSQL_DATABASE}" | tr '\t' '|' | while IFS='|' read IDQ IDEXERCICE IDTEAM REMOTE_USER SUBJECT CREATION STATE SOLVED CLOSED THEME EXERCICE_TITLE
|
||||||
|
do
|
||||||
|
# Search exercice by title
|
||||||
|
IDEXERCICE=$(echo "${EXERCICES}" | jq -r ".[] | select(.title == \"${EXERCICE_TITLE}\") | .id")
|
||||||
|
[ -z "$IDEXERCICE" ] && { >&2 echo "QA query $IDQ: no exercice match"; continue; }
|
||||||
|
|
||||||
|
# Search user in teams (if not found, set to null, this is allowed)
|
||||||
|
IDTEAM=$(echo "${TEAMS}" | jq -r "select(.login == \"${REMOTE_USER}\") | .id_team")
|
||||||
|
[ -z "$IDTEAM" ] && IDTEAM=null
|
||||||
|
|
||||||
|
CREATION=$(echo ${CREATION}Z | sed 's/ /T/')
|
||||||
|
[ "$SOLVED" == "NULL" ] && SOLVED=null || SOLVED="\"$(echo ${SOLVED}Z | sed 's/ /T/')\""
|
||||||
|
[ "$CLOSED" == "NULL" ] && CLOSED=null || CLOSED="\"$(echo ${CLOSED}Z | sed 's/ /T/')\""
|
||||||
|
|
||||||
|
SUBJECT=$(cat <<EOF | jq -R .
|
||||||
|
$SUBJECT
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
QA=$(curl -s -f -X POST -d @- "http://127.0.0.1:8081/api/qa/" <<EOF
|
||||||
|
{
|
||||||
|
"id_exercice": $IDEXERCICE,
|
||||||
|
"id_team": $IDTEAM,
|
||||||
|
"user": "$REMOTE_USER",
|
||||||
|
"creation": "$CREATION",
|
||||||
|
"state": "$STATE",
|
||||||
|
"subject": $SUBJECT,
|
||||||
|
"solved": $SOLVED,
|
||||||
|
"closed": $CLOSED
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
QAID=$(echo $QA | jq -r .id)
|
||||||
|
|
||||||
|
echo "SELECT * FROM qa_comments WHERE id_qa = $IDQ ORDER BY date ASC;" | mysql --skip-column-names -u "${MYSQL_USER}" --password="${MYSQL_PASSWORD}" "${MYSQL_DATABASE}" | tr '\t' '|' | while IFS='|' read IDC IDQ IDT REMOTE_USER DATEC COMMENT
|
||||||
|
do
|
||||||
|
|
||||||
|
IDTEAM=$(echo "${TEAMS}" | jq -r "select(.login == \"${REMOTE_USER}\") | .id_team")
|
||||||
|
[ -z "$IDTEAM" ] && IDTEAM=null
|
||||||
|
|
||||||
|
DATEC=$(echo ${DATEC}Z | sed 's/ /T/')
|
||||||
|
COMMENT=$(cat <<EOF | jq -R .
|
||||||
|
$COMMENT
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
curl -s -f -X POST -d @- "http://127.0.0.1:8081/api/qa/$QAID/comments" <<EOF > /dev/null
|
||||||
|
{
|
||||||
|
"id_team": $IDTEAM,
|
||||||
|
"user": "$REMOTE_USER",
|
||||||
|
"date": "$DATEC",
|
||||||
|
"content": $COMMENT
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
done
|
||||||
|
done
|
@ -42,9 +42,11 @@ func apiHandler(f DispatchFunction) func(http.ResponseWriter, *http.Request, htt
|
|||||||
} else if teamid, err = strconv.ParseInt(ficteam, 10, 64); err != nil {
|
} else if teamid, err = strconv.ParseInt(ficteam, 10, 64); err != nil {
|
||||||
if lnk, err := os.Readlink(path.Join(TeamsDir, ficteam)); err != nil {
|
if lnk, err := os.Readlink(path.Join(TeamsDir, ficteam)); err != nil {
|
||||||
log.Printf("[ERR] Unable to readlink %q: %s\n", path.Join(TeamsDir, ficteam), err)
|
log.Printf("[ERR] Unable to readlink %q: %s\n", path.Join(TeamsDir, ficteam), err)
|
||||||
|
http.Error(w, fmt.Sprintf("{errmsg:\"Unable to validate authentication.\"}"), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
} else if teamid, err = strconv.ParseInt(lnk, 10, 64); err != nil {
|
} else if teamid, err = strconv.ParseInt(lnk, 10, 64); err != nil {
|
||||||
log.Printf("[ERR] Error during ParseInt team %q: %s\n", lnk, err)
|
log.Printf("[ERR] Error during ParseInt team %q: %s\n", lnk, err)
|
||||||
|
http.Error(w, fmt.Sprintf("{errmsg:\"Unable to validate authentication.\"}"), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -74,7 +74,7 @@ func createExerciceQA(u QAUser, exercice fic.Exercice, body []byte) (interface{}
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if qa, err := exercice.NewQAQuery(uq.Subject, u.TeamId, u.User, uq.State); err != nil {
|
if qa, err := exercice.NewQAQuery(uq.Subject, &u.TeamId, u.User, uq.State); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
} else {
|
} else {
|
||||||
var uc fic.QAComment
|
var uc fic.QAComment
|
||||||
@ -83,7 +83,7 @@ func createExerciceQA(u QAUser, exercice fic.Exercice, body []byte) (interface{}
|
|||||||
}
|
}
|
||||||
|
|
||||||
if uc.Content != "" {
|
if uc.Content != "" {
|
||||||
_, err = qa.AddComment(uc.Content, u.TeamId, u.User)
|
_, err = qa.AddComment(uc.Content, &u.TeamId, u.User)
|
||||||
}
|
}
|
||||||
|
|
||||||
return qa, err
|
return qa, err
|
||||||
@ -132,7 +132,7 @@ func createQAComment(u QAUser, query fic.QAQuery, exercice fic.Exercice, body []
|
|||||||
return nil, errors.New("Empty comment")
|
return nil, errors.New("Empty comment")
|
||||||
}
|
}
|
||||||
|
|
||||||
return query.AddComment(uc.Content, u.TeamId, u.User)
|
return query.AddComment(uc.Content, &u.TeamId, u.User)
|
||||||
}
|
}
|
||||||
|
|
||||||
func deleteQAComment(u QAUser, comment fic.QAComment, query fic.QAQuery, exercice fic.Exercice, body []byte) (interface{}, error) {
|
func deleteQAComment(u QAUser, comment fic.QAComment, query fic.QAQuery, exercice fic.Exercice, body []byte) (interface{}, error) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user