teachers.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. package api
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "html/template"
  6. "io/ioutil"
  7. "net/http"
  8. "sort"
  9. "strings"
  10. "github.com/gorilla/mux"
  11. "github.com/gorilla/schema"
  12. "gogs.carducci-dante.gov.it/karmen/core/renderer"
  13. "gogs.carducci-dante.gov.it/karmen/datasource/edt"
  14. "gogs.carducci-dante.gov.it/karmen/orm"
  15. )
  16. var (
  17. selectUniqueTeacherClasses = `
  18. SELECT name FROM (SELECT activities.id,classes.name FROM activities
  19. INNER JOIN classes on classes.id=activities.class_id
  20. WHERE teacher_id=?) as grouped GROUP BY name`
  21. selectUniqueTeacherSubjects = `
  22. SELECT name FROM (SELECT activities.id,subjects.name FROM activities
  23. INNER JOIN subjects on subjects.id=activities.subject_id
  24. WHERE activities.teacher_id=?) as grouped GROUP BY name`
  25. )
  26. func teachersHandler() http.Handler {
  27. fn := func(w http.ResponseWriter, r *http.Request) {
  28. var teachers []*orm.Teacher
  29. if err := orm.DB().Find(&teachers).Error; err != nil {
  30. panic(err)
  31. }
  32. for _, teacher := range teachers {
  33. orm.DB().Raw(selectUniqueTeacherClasses, teacher.ID).Scan(&teacher.Classes)
  34. orm.DB().Raw(selectUniqueTeacherSubjects, teacher.ID).Scan(&teacher.Subjects)
  35. }
  36. sort.Slice(teachers, func(i, j int) bool {
  37. return strings.ToLower(teachers[i].Surname) < strings.ToLower(teachers[j].Surname)
  38. })
  39. if err := renderer.Render(w, "base", "teachers", teachers); err != nil {
  40. panic(err)
  41. }
  42. }
  43. return http.HandlerFunc(fn)
  44. }
  45. func teacherShowHandler() http.Handler {
  46. fn := func(w http.ResponseWriter, r *http.Request) {
  47. var data struct {
  48. Teacher orm.Teacher
  49. Subjects []orm.Subject
  50. Classes []orm.Class
  51. Activities []orm.Activity
  52. }
  53. if err := orm.DB().First(&data.Teacher, mux.Vars(r)["id"]).Error; err != nil {
  54. panic(err)
  55. }
  56. if err := orm.DB().Model(&data.Teacher).Related(&data.Subjects, "Subjects").Error; err != nil {
  57. panic(err)
  58. }
  59. if err := orm.DB().Model(&data.Teacher).Related(&data.Classes, "Classes").Error; err != nil {
  60. panic(err)
  61. }
  62. sort.Slice(data.Classes, func(i, j int) bool {
  63. return strings.ToLower(data.Classes[i].Name) < strings.ToLower(data.Classes[j].Name)
  64. })
  65. if err := orm.DB().Model(&data.Teacher).Related(&data.Activities, "Activities").Error; err != nil {
  66. panic(err)
  67. }
  68. sort.Slice(data.Activities, func(i, j int) bool {
  69. return data.Activities[i].Hours > data.Activities[j].Hours
  70. })
  71. if err := renderer.Render(w, "base", "teachers_show", data); err != nil {
  72. panic(err)
  73. }
  74. }
  75. return http.HandlerFunc(fn)
  76. }
  77. func teachersAddHandler() http.Handler {
  78. fn := func(w http.ResponseWriter, r *http.Request) {
  79. if r.Method == "GET" {
  80. var data struct {
  81. Update bool
  82. SubmitUrl template.URL
  83. }
  84. data.SubmitUrl = template.URL(fmt.Sprintf("/teachers/add"))
  85. if err := renderer.Render(w, "base", "teachers_add_update", data); err != nil {
  86. panic(err)
  87. }
  88. } else if r.Method == "POST" {
  89. var teacher orm.Teacher
  90. if err := r.ParseForm(); err != nil {
  91. panic(err)
  92. }
  93. decoder := schema.NewDecoder()
  94. if err := decoder.Decode(&teacher, r.PostForm); err != nil {
  95. panic(err)
  96. }
  97. // Generate username and initial password
  98. teacher.Username = teacher.GenerateUsername()
  99. teacher.Password = teacher.GenerateSaltedPassword(teacher.Username)
  100. orm.DB().NewRecord(teacher)
  101. if err := orm.DB().Create(&teacher).Error; err != nil {
  102. panic(err)
  103. }
  104. http.Redirect(w, r, fmt.Sprintf("/teachers/%d", teacher.ID), http.StatusSeeOther)
  105. }
  106. }
  107. return http.HandlerFunc(fn)
  108. }
  109. func teachersDeleteHandler() http.Handler {
  110. fn := func(w http.ResponseWriter, r *http.Request) {
  111. var teacher orm.Teacher
  112. if err := orm.DB().First(&teacher, mux.Vars(r)["id"]).Error; err != nil {
  113. panic(err)
  114. }
  115. if err := orm.DB().Delete(&teacher).Error; err != nil {
  116. panic(err)
  117. }
  118. var data struct {
  119. RedirectUrl string `json:"redirect_url"`
  120. }
  121. data.RedirectUrl = "/teachers"
  122. w.Header().Set("Content-Type", "application/json")
  123. json.NewEncoder(w).Encode(data)
  124. }
  125. return http.HandlerFunc(fn)
  126. }
  127. func teacherUpdateHandler() http.Handler {
  128. fn := func(w http.ResponseWriter, r *http.Request) {
  129. if r.Method == "GET" {
  130. var data struct {
  131. Update bool
  132. SubmitUrl template.URL
  133. Teacher orm.Teacher
  134. }
  135. if err := orm.DB().First(&data.Teacher, mux.Vars(r)["id"]).Error; err != nil {
  136. panic(err)
  137. }
  138. data.Update = true
  139. data.SubmitUrl = template.URL(fmt.Sprintf("/teachers/%d/update", data.Teacher.ID))
  140. if err := renderer.Render(w, "base", "teachers_add_update", data); err != nil {
  141. panic(err)
  142. }
  143. } else if r.Method == "POST" {
  144. var (
  145. teacher orm.Teacher
  146. oldPassword string
  147. )
  148. if err := orm.DB().First(&teacher, mux.Vars(r)["id"]).Error; err != nil {
  149. panic(err)
  150. }
  151. oldPassword = teacher.Password
  152. if err := r.ParseForm(); err != nil {
  153. panic(err)
  154. }
  155. decoder := schema.NewDecoder()
  156. if err := decoder.Decode(&teacher, r.PostForm); err != nil {
  157. panic(err)
  158. }
  159. if oldPassword != teacher.Password {
  160. teacher.Password = teacher.GenerateSaltedPassword(teacher.Password)
  161. }
  162. if err := orm.DB().Save(&teacher).Error; err != nil {
  163. panic(err)
  164. }
  165. var data struct {
  166. Teachers []orm.Teacher
  167. }
  168. data.Teachers = make([]orm.Teacher, 0)
  169. if err := orm.DB().Find(&data.Teachers).Error; err != nil {
  170. panic(err)
  171. }
  172. http.Redirect(w, r, fmt.Sprintf("/teachers/%d", teacher.ID), http.StatusSeeOther)
  173. }
  174. }
  175. return http.HandlerFunc(fn)
  176. }
  177. func teachersCheckUsernameHandler() http.Handler {
  178. fn := func(w http.ResponseWriter, r *http.Request) {
  179. var teacher orm.Teacher
  180. var data struct {
  181. Exists bool `json:"exists"`
  182. }
  183. if orm.DB().Where("username = ?", mux.Vars(r)["username"]).Find(&teacher).RecordNotFound() {
  184. data.Exists = false
  185. } else {
  186. data.Exists = true
  187. }
  188. w.Header().Set("Content-Type", "application/json")
  189. json.NewEncoder(w).Encode(data)
  190. }
  191. return http.HandlerFunc(fn)
  192. }
  193. func teachersImportHandler() http.Handler {
  194. fn := func(w http.ResponseWriter, r *http.Request) {
  195. r.ParseMultipartForm(32 << 20)
  196. file, _, err := r.FormFile("teachers_import")
  197. if err != nil {
  198. panic(err)
  199. }
  200. defer file.Close()
  201. data, err := ioutil.ReadAll(file)
  202. if err != nil {
  203. panic(err)
  204. }
  205. if importer, err := edt.NewImporter(
  206. string(data),
  207. map[string]string{
  208. "COGNOME": "surname",
  209. "NOME": "name",
  210. },
  211. ); err != nil {
  212. panic(err)
  213. } else {
  214. if err := orm.Import(&orm.Teacher{}, importer); err != nil {
  215. panic(err)
  216. }
  217. http.Redirect(w, r, "/teachers", http.StatusSeeOther)
  218. }
  219. }
  220. return http.HandlerFunc(fn)
  221. }