diff --git a/internal/models/area.go b/internal/models/area.go
index 31f0af9e49cf231fb734b19ca2345623d77ad6c4..cd66e003060caaadbdb4bfd74f5e85b4dec72d04 100644
--- a/internal/models/area.go
+++ b/internal/models/area.go
@@ -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)
diff --git a/internal/models/election.go b/internal/models/election.go
index f53f6e73935cbe0be438b53c318e325b4f402efb..970cfd4330c6d942d71bc9049401ce18f2d66f7c 100644
--- a/internal/models/election.go
+++ b/internal/models/election.go
@@ -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)
diff --git a/internal/models/models.go b/internal/models/models.go
index 40fc52ba7df3ef9e4c7742d7f6a4d9b7bbb96d3a..38eb6d2315618ae594525a3bd73c03f629a2138a 100644
--- a/internal/models/models.go
+++ b/internal/models/models.go
@@ -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"`
diff --git a/internal/models/section.go b/internal/models/section.go
index dd81f8df94f8ad0fa5aedd98486998c611fc8b5e..3e9ec600601d14db42125fd5b1d5896c1559a332 100644
--- a/internal/models/section.go
+++ b/internal/models/section.go
@@ -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)
diff --git a/internal/rootmux/rootmux_test.go b/internal/rootmux/rootmux_test.go
index c98e56807570f073af224f38e7f393fbc4a11eee..9ec0ac68dbd79e121e6fc838b2f342d68ddf0422 100644
--- a/internal/rootmux/rootmux_test.go
+++ b/internal/rootmux/rootmux_test.go
@@ -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>")