Skip to content
Snippets Groups Projects
Commit d5118d36 authored by Alexis POYEN's avatar Alexis POYEN
Browse files

Merge branch '18-deletion-in-cascade' into 'master'

Resolve "Deletion in Cascade"

Closes #18

See merge request apoyen/elections!12
parents 3b3119d8 93a71550
No related branches found
No related tags found
1 merge request!12Resolve "Deletion in Cascade"
Pipeline #5447 passed
......@@ -110,10 +110,15 @@ func (d *DataHandler) putArea(w http.ResponseWriter, r *http.Request, id int) {
func (d *DataHandler) deleteArea(w http.ResponseWriter, r *http.Request, id int) {
if id != 0 {
var o Area
if err := d.db.First(&o, id).Error; err != nil {
if err := d.db.Preload("Sections").First(&o, id).Error; err != nil {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
return
}
for _, section := range o.Sections {
d.deleteSection(w, r, int(section.ID))
}
d.db.Delete(&o)
} else {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
......
......@@ -15,14 +15,14 @@ func (d *DataHandler) handleElection(w http.ResponseWriter, r *http.Request) {
case "GET":
switch auth.GetLoggedUserTechnical(w, r).Role {
case "ADMIN", "CAPTURER", "VISUALIZER":
d.getElectionAdmin(w, r, id)
d.getElection(w, r, id)
default:
http.Error(w, ErrorRoleOfLoggedUser, http.StatusInternalServerError)
}
case "POST":
switch auth.GetLoggedUserTechnical(w, r).Role {
case "ADMIN":
d.postElectionAdmin(w, r)
d.postElection(w, r)
case "CAPTURER", "VISUALIZER":
http.Error(w, ErrorNotAuthorizeMethodOnRessource, http.StatusMethodNotAllowed)
default:
......@@ -32,7 +32,7 @@ func (d *DataHandler) handleElection(w http.ResponseWriter, r *http.Request) {
case "PUT":
switch auth.GetLoggedUserTechnical(w, r).Role {
case "ADMIN":
d.putElectionAdmin(w, r, id)
d.putElection(w, r, id)
case "CAPTURER", "VISUALIZER":
http.Error(w, ErrorNotAuthorizeMethodOnRessource, http.StatusMethodNotAllowed)
default:
......@@ -41,7 +41,7 @@ func (d *DataHandler) handleElection(w http.ResponseWriter, r *http.Request) {
case "DELETE":
switch auth.GetLoggedUserTechnical(w, r).Role {
case "ADMIN":
d.deleteElectionAdmin(w, r, id)
d.deleteElection(w, r, id)
case "CAPTURER", "VISUALIZER":
http.Error(w, ErrorNotAuthorizeMethodOnRessource, http.StatusMethodNotAllowed)
default:
......@@ -52,7 +52,7 @@ func (d *DataHandler) handleElection(w http.ResponseWriter, r *http.Request) {
}
}
func (d *DataHandler) getElectionAdmin(w http.ResponseWriter, r *http.Request, id int) {
func (d *DataHandler) getElection(w http.ResponseWriter, r *http.Request, id int) {
if id != 0 {
var o Election
if err := d.db.Preload("Areas").First(&o, id).Error; err != nil {
......@@ -67,7 +67,7 @@ func (d *DataHandler) getElectionAdmin(w http.ResponseWriter, r *http.Request, i
}
}
func (d *DataHandler) postElectionAdmin(w http.ResponseWriter, r *http.Request) {
func (d *DataHandler) postElection(w http.ResponseWriter, r *http.Request) {
var o Election
err := json.NewDecoder(r.Body).Decode(&o)
if err != nil {
......@@ -80,7 +80,7 @@ func (d *DataHandler) postElectionAdmin(w http.ResponseWriter, r *http.Request)
}
func (d *DataHandler) putElectionAdmin(w http.ResponseWriter, r *http.Request, id int) {
func (d *DataHandler) putElection(w http.ResponseWriter, r *http.Request, id int) {
var o Election
if err := d.db.Preload("Areas").First(&o, id).Error; err != nil {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
......@@ -99,13 +99,18 @@ func (d *DataHandler) putElectionAdmin(w http.ResponseWriter, r *http.Request, i
}
func (d *DataHandler) deleteElectionAdmin(w http.ResponseWriter, r *http.Request, id int) {
func (d *DataHandler) deleteElection(w http.ResponseWriter, r *http.Request, id int) {
if id != 0 {
var o Election
if err := d.db.First(&o, id).Error; err != nil {
if err := d.db.Preload("Areas").First(&o, id).Error; err != nil {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
return
}
for _, area := range o.Areas {
d.deleteArea(w, r, int(area.ID))
}
d.db.Delete(&o)
} else {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
......
......@@ -35,7 +35,7 @@ const ErrorNotAuthorizeMethodOnRessource = "You're not authorize to execute this
// ErrorParentNotFound = "Could not get the parent associated to the object" with 500 http.StatusInternalServerError
const ErrorParentNotFound = "Could not get the parent associated to the object"
// Election
// Election represent an election divided in areas with 1 or several rounds
type Election struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -47,7 +47,7 @@ type Election struct {
Rounds []Round
}
// Area
// Area represent an area of an election divided in one or several Sections
type Area struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -60,7 +60,7 @@ type Area struct {
Sections []Section
}
// Section
// Section represent a section of an area divided in 1 or several Desks
type Section struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -72,7 +72,7 @@ type Section struct {
Desks []Desk
}
// Desk
// Desk represent a Desk office to vote from a section with the number of subscribed. It can be set to be a witness desk
type Desk struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -84,7 +84,7 @@ type Desk struct {
Subscribed uint
}
// Party
// Party represent a political party or tendance
type Party struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -95,7 +95,7 @@ type Party struct {
CandidateList []CandidateList
}
// Capturer
// Capturer is a user who can capture the results on the desks he is affected
type Capturer struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -106,7 +106,7 @@ type Capturer struct {
DeskRounds []DeskRound `gorm:"many2many:capturer_deskrounds;"`
}
// Parameter
// Parameter save the parameter for a round
type Parameter struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -117,7 +117,7 @@ type Parameter struct {
ShowMap bool
}
// Round
// Round represent a round for an election
type Round struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -132,7 +132,7 @@ type Round struct {
CandidateLists []CandidateList
}
// DeskRound
// DeskRound is a duplicate instance of a Desk to save the result for a round
type DeskRound struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -146,7 +146,7 @@ type DeskRound struct {
Votes []Vote
}
// CandidateList
// CandidateList is a list presented in an Area on an election
type CandidateList struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -160,7 +160,7 @@ type CandidateList struct {
Votes []Vote
}
// Candidate
// Candidate is a candiate presented on a list
type Candidate struct {
ID uint `gorm:"primary_key"`
CreatedAt time.Time `json:"-"`
......@@ -176,7 +176,7 @@ type Candidate struct {
Removed bool
}
// Vote
// Vote represent the number of voice between a CanidateList and a Desk (+blank and null)
type Vote struct {
DeskRoundID uint `gorm:"primary_key"`
CandidateListID uint `gorm:"primary_key"`
......
......@@ -109,10 +109,15 @@ func (d *DataHandler) putSection(w http.ResponseWriter, r *http.Request, id int)
func (d *DataHandler) deleteSection(w http.ResponseWriter, r *http.Request, id int) {
if id != 0 {
var o Section
if err := d.db.First(&o, id).Error; err != nil {
if err := d.db.Preload("Desks").First(&o, id).Error; err != nil {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
return
}
for _, desk := range o.Desks {
d.deleteDesk(w, r, int(desk.ID))
}
d.db.Delete(&o)
} else {
http.Error(w, ErrorIDIsMissing, http.StatusNotFound)
......
......@@ -87,7 +87,20 @@ func appTests(t *testing.T) {
json.Unmarshal([]byte(response), &token)
xsrfHeader := tester.Header{Key: "XSRF-TOKEN", Value: token.XSRFToken}
// Add a capturer to an already bind UserID should fail
do("POST", "/api/Capturer", xsrfHeader, `{"UserID":2,"Name":"Capturer"}`, 500, `UserID is already bind to a Capturer`)
// Test deletion in cascade for generic election
do("POST", "/api/Election", xsrfHeader, `{"Name":"Grand Lyon 2020","BallotType":"metropolitan-direct"}`, 200, `{"ID":1,"Name":"Grand Lyon 2020","BallotType":"metropolitan-direct","Areas":null,"Rounds":null}`)
do("POST", "/api/Area", xsrfHeader, `{"ElectionID":1,"Name":"Area 1","SeatNumber":9,"MapID":"1"}`, 200, `{"ID":1,"ElectionID":1,"Name":"Area 1","SeatNumber":9,"MapID":"1","Sections":null}`)
do("POST", "/api/Section", xsrfHeader, `{"AreaID":1,"Name":"Section 1","MapID":"1"}`, 200, `{"ID":1,"AreaID":1,"Name":"Section 1","MapID":"1","Desks":null}`)
do("POST", "/api/Desk", xsrfHeader, `{"SectionID":1,"Name":"Desk 1","WitnessDesk":true,"Subscribed":9587}`, 200, `{"ID":1,"SectionID":1,"Name":"Desk 1","WitnessDesk":true,"Subscribed":9587}`)
do("DELETE", "/api/Election/1", xsrfHeader, ``, 200, ``)
do("GET", "/api/Area/1", xsrfHeader, ``, 404, `id is missing`)
do("GET", "/api/Section/1", xsrfHeader, ``, 404, `id is missing`)
do("GET", "/api/Desk/1", xsrfHeader, ``, 404, `id is missing`)
}
// Do an OAuth2 login with an known admin
do("GET", "/OAuth2Login", noH, "", 200, "<!DOCTYPE html>")
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment