main.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  1. package main
  2. import (
  3. "fmt"
  4. "github.com/boombuler/barcode"
  5. "github.com/boombuler/barcode/qr"
  6. "github.com/gorilla/mux"
  7. "gorm.io/driver/sqlite"
  8. "gorm.io/gorm"
  9. "html/template"
  10. "image"
  11. "image/png"
  12. "io/ioutil"
  13. "math/rand"
  14. "net/http"
  15. "os"
  16. "strconv"
  17. "strings"
  18. "time"
  19. )
  20. func init() {
  21. rand.Seed(time.Now().UnixNano())
  22. }
  23. var letterRunes = []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890")
  24. func main() {
  25. // port := os.Getenv("PORT")
  26. r := mux.NewRouter()
  27. r.HandleFunc("/", PrinterList)
  28. r.HandleFunc("/generate", GenerateQR)
  29. r.HandleFunc("/generateCompatible", generateCompatible)
  30. r.HandleFunc("/addprinter", AddPrinter)
  31. r.HandleFunc("/addcartridge", AddCartridge)
  32. r.HandleFunc("/cartridgeOfPrinter", AddCartridgeOfPrinter)
  33. r.HandleFunc("/compatible", FindCompatibleCartridges)
  34. r.HandleFunc("/printer/{printerName}", PrinterPage)
  35. r.HandleFunc("/cartridges", CartridgePage)
  36. r.HandleFunc("/updateCartridge", updateCartridges)
  37. r.HandleFunc("/decreaseCartridgeQuantity", decreaseCartridgeQuantity)
  38. fmt.Println("Server is listening...")
  39. http.Handle("/", r)
  40. http.ListenAndServe(":80" /*+port*/, nil)
  41. }
  42. func decreaseCartridgeQuantity(w http.ResponseWriter, r *http.Request) {
  43. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  44. r.ParseForm()
  45. selectedCartridge := strings.Join(r.Form["cartridge"], "")
  46. var cartridge Cartridges
  47. db.Where("Name = ?", selectedCartridge).First(&cartridge)
  48. if cartridge.Quantity > 0 {
  49. newQuantity := cartridge.Quantity - 1
  50. db.Model(&cartridge).Update("Quantity", newQuantity)
  51. } else {
  52. fmt.Sprint(w, "Картридж закончился")
  53. }
  54. http.Redirect(w, r, "/", 301)
  55. }
  56. func updateCartridges(w http.ResponseWriter, r *http.Request) {
  57. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  58. r.ParseForm()
  59. selectedCartridge := strings.Join(r.Form["cartridges"], "")
  60. var cartridge Cartridges
  61. db.Where("Name = ?", selectedCartridge).First(&cartridge)
  62. quantity, err := strconv.Atoi(strings.Join(r.Form["cartridgeQuantity"], ""))
  63. if err != nil {
  64. fmt.Println("Someone tries to update int value by string. Someone: " + r.RemoteAddr)
  65. }
  66. newQuantity := cartridge.Quantity + uint(quantity)
  67. db.Model(&cartridge).Update("Quantity", newQuantity)
  68. http.Redirect(w, r, "/", 301)
  69. }
  70. func generateCompatible(w http.ResponseWriter, r *http.Request) {
  71. fmt.Println("New generating request from" + r.RemoteAddr)
  72. data, err := ioutil.ReadAll(r.Body)
  73. if err != nil {
  74. fmt.Println("Something went wrong during generating QR code")
  75. fmt.Fprintf(w, "Something went wrong during generating QR code")
  76. }
  77. filename := GenerateRandomString(10)
  78. fmt.Println("Generated name " + filename + " for " + r.RemoteAddr)
  79. generateFromText(string(data), filename)
  80. fmt.Println("Generated file " + filename + ".png for " + r.RemoteAddr)
  81. w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", filename+".png"))
  82. http.ServeFile(w, r, filename)
  83. fmt.Println("Served file " + filename + ".png for " + r.RemoteAddr)
  84. os.Remove(filename + ".png")
  85. fmt.Println("Removed file " + filename + ".png")
  86. http.Redirect(w, r, "/", 301)
  87. }
  88. func CartridgePage(w http.ResponseWriter, r *http.Request) {
  89. db, err := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  90. if err != nil {
  91. fmt.Println("Error during opening DB")
  92. } else {
  93. var cartridges []Cartridges
  94. var names []string
  95. db.Find(&cartridges)
  96. for _, v := range cartridges {
  97. qua := fmt.Sprint(v.Quantity)
  98. name := v.Name + ": " + qua
  99. names = append(names, name)
  100. }
  101. data := CartridgesOutput{
  102. Name: names,
  103. }
  104. tmpl, err := template.ParseFiles("cartridges.gohtml")
  105. if err != nil {
  106. fmt.Println("Error parsing files")
  107. fmt.Println(err)
  108. }
  109. tmpl.Execute(w, data)
  110. }
  111. }
  112. func PrinterPage(w http.ResponseWriter, r *http.Request) {
  113. db, err := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  114. if err != nil {
  115. fmt.Println("Error during opening DB")
  116. } else {
  117. name := mux.Vars(r)["printerName"]
  118. var printer Printers
  119. db.Where("Name = ?", name).First(&printer)
  120. var printers []Printers
  121. db.Find(&printers)
  122. fmt.Println(printer)
  123. if printer.ID > 0 {
  124. var cartridges []string
  125. var cot []Cartridgeofprinter
  126. db.Find(&cot)
  127. db.Where("Printer_Id = ?", printer.ID).Find(&cot)
  128. fmt.Println(cot)
  129. for _, v := range cot {
  130. var cart Cartridges
  131. if v.Printer_Id == printer.ID {
  132. db.Where("ID = ?", v.Cartridge_Id).First(&cart)
  133. if cart.Quantity > 0 {
  134. cartridges = append(cartridges, cart.Name)
  135. }
  136. }
  137. }
  138. data := Output{
  139. PrinterName: name,
  140. Cartridges: cartridges,
  141. }
  142. tmpl, _ := template.ParseFiles("output.html")
  143. tmpl.Execute(w, data)
  144. } else {
  145. http.Redirect(w, r, "https://www.youtube.com/watch?v=dQw4w9WgXcQ", 301)
  146. }
  147. }
  148. // w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", name + ".png"))
  149. // var cartText = strings.Join(cartridgeName, "\n")
  150. // cartridgeQR(cartText, name)
  151. // http.ServeFile(w, r, name)
  152. }
  153. func FindCompatibleCartridges(w http.ResponseWriter, r *http.Request) {
  154. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  155. var PrinterList []Printers
  156. db.Find(&PrinterList)
  157. var PrinterName []string
  158. for _, pr := range PrinterList {
  159. PrinterName = append(PrinterName, pr.Name)
  160. }
  161. // var printer Printers
  162. r.ParseForm()
  163. selectedPrinter := strings.Join(r.Form["printer"], "")
  164. if selectedPrinter != "" {
  165. // db.Where("Name = ?", selectedPrinter).First(&printer)
  166. // var cops []uint
  167. // db.Table("CartridgeOfPrinters").Where("PrinterID = ?", printer.ID).Select("CartridgeID").Find(&cops)
  168. // var cartridgeName []string
  169. // for _, cop := range cops {
  170. // var cart Cartridges
  171. // db.Where("ID = ?", cop).First(&cart)
  172. // cartridgeName = append(cartridgeName, cart.Name)
  173. // }
  174. // // data := ViewData{
  175. // // Title: "Совместимость картриджей",
  176. // // PrinterNames: PrinterName,
  177. // // CartridgeNames: cartridgeName,
  178. // // }
  179. http.Redirect(w, r, "/printer/"+selectedPrinter, 301)
  180. return
  181. }
  182. data := ViewData{
  183. Title: "Совместимость картриджей",
  184. PrinterNames: PrinterName,
  185. }
  186. tmpl, _ := template.ParseFiles("compatible.html")
  187. tmpl.Execute(w, data)
  188. }
  189. func AddCartridgeOfPrinter(w http.ResponseWriter, r *http.Request) {
  190. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  191. r.ParseForm()
  192. cartridgeName := strings.Join(r.Form["cartridges"], "")
  193. printerName := strings.Join(r.Form["printers"], "")
  194. // printerName := strings.Join(r.Form["printers"], "")
  195. var cartridge Cartridges
  196. db.Where("Name = ?", cartridgeName).First(&cartridge)
  197. cartridgeID := cartridge.ID
  198. var printer Printers
  199. db.Where("Name = ?", printerName).First(&printer)
  200. printerID := printer.ID
  201. var cop Cartridgeofprinter
  202. cop.Cartridge_Id = cartridgeID
  203. cop.Printer_Id = printerID
  204. db.Create(&cop)
  205. http.Redirect(w, r, "/", 302)
  206. }
  207. func AddCartridge(w http.ResponseWriter, r *http.Request) {
  208. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  209. r.ParseForm()
  210. text := strings.Join(r.Form["cartridgeName"], "")
  211. var newCartridge Cartridges
  212. newCartridge.Name = text
  213. db.Create(&newCartridge)
  214. http.Redirect(w, r, "/", 302)
  215. }
  216. func AddPrinter(w http.ResponseWriter, r *http.Request) {
  217. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  218. r.ParseForm()
  219. text := strings.Join(r.Form["printerName"], "")
  220. var newPrinter Printers
  221. newPrinter.Name = text
  222. db.Create(&newPrinter)
  223. http.Redirect(w, r, "/", 302)
  224. }
  225. func GenerateQR(w http.ResponseWriter, r *http.Request) {
  226. r.ParseForm() // Получение имени принтера
  227. text := strings.Join(r.Form["printer"], "")
  228. GenerateFromWeb(w, r, text)
  229. }
  230. func GenerateFromWeb(w http.ResponseWriter, r *http.Request, printerName string) {
  231. qrtext := "https://printers-ttit.herokuapp.com/printer/" + printerName // Генерация URL
  232. fmt.Println("Generating QR code with text: " + qrtext + " for " + r.RemoteAddr)
  233. filename := GenerateRandomString(10) // Генерация имени файла
  234. fmt.Println("Generated filename " + filename + " for " + r.RemoteAddr)
  235. err := generateFromText(qrtext, filename) // qrtext - текст в QR, filename - имя файла
  236. if err != nil {
  237. fmt.Println(err)
  238. }
  239. fmt.Println("Setting header Content-Disposition for " + r.RemoteAddr)
  240. w.Header().Set("Content-Disposition", fmt.Sprintf("attachment; filename=\"%s\"", filename+".png"))
  241. fmt.Println("Starting serving file for " + r.RemoteAddr)
  242. http.ServeFile(w, r, filename+".png")
  243. }
  244. func generateFromText(text string, filename string) error {
  245. code, err := qr.Encode(text, qr.L, qr.Auto)
  246. if err != nil {
  247. return fmt.Errorf("Error during generating QR code")
  248. }
  249. if text != code.Content() {
  250. return fmt.Errorf("data differs")
  251. }
  252. code, err = barcode.Scale(code, 512, 512)
  253. if err != nil {
  254. return fmt.Errorf("Error during scaling QR code")
  255. }
  256. err = writePng(filename, code)
  257. if err != nil {
  258. return err
  259. }
  260. return nil
  261. }
  262. func writePng(filename string, img image.Image) error {
  263. file, err := os.Create(filename + ".png")
  264. if err != nil {
  265. return fmt.Errorf("Creating file error")
  266. }
  267. err = png.Encode(file, img)
  268. if err != nil {
  269. return fmt.Errorf("PNG file error")
  270. }
  271. file.Close()
  272. return nil
  273. }
  274. func GenerateRandomString(n int) string {
  275. b := make([]rune, n)
  276. for i := range b {
  277. b[i] = letterRunes[rand.Intn(len(letterRunes))]
  278. }
  279. return string(b)
  280. }
  281. func PrinterList(w http.ResponseWriter, r *http.Request) {
  282. db, _ := gorm.Open(sqlite.Open("printer.db"), &gorm.Config{})
  283. var PrinterList []Printers
  284. db.Find(&PrinterList)
  285. var PrinterName []string
  286. for _, pr := range PrinterList {
  287. PrinterName = append(PrinterName, pr.Name)
  288. }
  289. var CartridgeList []Cartridges
  290. db.Find(&CartridgeList)
  291. var CartridgeName []string
  292. for _, ca := range CartridgeList {
  293. CartridgeName = append(CartridgeName, ca.Name)
  294. }
  295. data := ViewData{
  296. Title: "Generate QR",
  297. PrinterNames: PrinterName,
  298. CartridgeNames: CartridgeName,
  299. }
  300. tmpl, _ := template.ParseFiles("generate.html")
  301. tmpl.Execute(w, data)
  302. }
  303. func cartridgeQR(text string, filename string) {
  304. code, err := qr.Encode(text, qr.L, qr.Auto)
  305. if err != nil {
  306. fmt.Println("Something went wrong...")
  307. }
  308. if text != code.Content() {
  309. panic("data differs")
  310. }
  311. code, err = barcode.Scale(code, 512, 512)
  312. if err != nil {
  313. panic(err)
  314. }
  315. writePng(filename, code)
  316. }