package controllers import ( "bytes" "encoding/json" "fmt" "net/http" "os" "time" "s-class-backend/config" // Import database mqtt "github.com/eclipse/paho.mqtt.golang" "github.com/gin-gonic/gin" ) // --- STRUKTUR DATA REQUEST --- type DeviceControlRequest struct { Device string `json:"device"` Action string `json:"action"` } type VerifyRequest struct { Token string `json:"token"` } // ========================================================================= // CACHE STATUS HARDWARE (Mengingat status terakhir perangkat untuk Web Admin) // ========================================================================= var DeviceStatusCache = map[string]string{ "lampu1": "off", "lampu2": "off", "ac": "off", "projector": "off", } // ========================================================================= // FUNGSI 1: VERIFIKASI TOKEN (Dan Buka Gembok Relay) // ========================================================================= func VerifyHardwareCode(c *gin.Context) { var req VerifyRequest if err := c.ShouldBindJSON(&req); err != nil { c.JSON(http.StatusBadRequest, gin.H{"error": "Format token tidak valid"}) return } tokenInput := req.Token sekarang := time.Now() // 1. Token Master CS (Cleaning Service) if tokenInput == "CS2026" { fmt.Printf("[VERIFY] Master Token CS digunakan\n") c.JSON(http.StatusOK, gin.H{ "status": "success", "message": "Token CS Valid", "duration_minutes": 60, }) return } // 2. Token Master Admin if tokenInput == "ADM999" { fmt.Printf("[VERIFY] Master Token Admin digunakan\n") c.JSON(http.StatusOK, gin.H{ "status": "success", "message": "Token Admin Valid", "duration_minutes": 999, }) return } // ------------------------------------------------------------- // TARIK DATA START_TIME & END_TIME MENGGUNAKAN time.Time // ------------------------------------------------------------- var jamMulai time.Time var jamSelesai time.Time isTokenValid := false type ResultTime struct { JamMulai time.Time `gorm:"column:start_time"` JamSelesai time.Time `gorm:"column:end_time"` } var result ResultTime errBooking := config.DB.Table("bookings"). Select("start_time, end_time"). Where("redeem_code = ?", tokenInput). Scan(&result).Error if errBooking == nil && !result.JamSelesai.IsZero() { jamMulai = result.JamMulai jamSelesai = result.JamSelesai isTokenValid = true } else { errSchedule := config.DB.Table("class_schedules"). Select("start_time, end_time"). Where("kode_mk = ?", tokenInput). Scan(&result).Error if errSchedule == nil && !result.JamSelesai.IsZero() { jamMulai = result.JamMulai jamSelesai = result.JamSelesai isTokenValid = true } } if !isTokenValid { c.JSON(http.StatusUnauthorized, gin.H{"error": "Token salah atau tidak ditemukan"}) return } // Ekstrak Jam, Menit, Detik langsung dari objek Time (Sangat Aman & Akurat) jamMulaiHariIni := time.Date(sekarang.Year(), sekarang.Month(), sekarang.Day(), jamMulai.Hour(), jamMulai.Minute(), jamMulai.Second(), 0, sekarang.Location()) jamSelesaiHariIni := time.Date(sekarang.Year(), sekarang.Month(), sekarang.Day(), jamSelesai.Hour(), jamSelesai.Minute(), jamSelesai.Second(), 0, sekarang.Location()) // ------------------------------------------------------------- // LOGIKA PENOLAKAN TOKEN MASA DEPAN // ------------------------------------------------------------- // Toleransi masuk: Hanya bisa masuk 15 menit sebelum jadwal. batasMasukAwal := jamMulaiHariIni.Add(-15 * time.Minute) if sekarang.Before(batasMasukAwal) { fmt.Printf("[VERIFY] Ditolak: Kelas %s belum mulai\n", tokenInput) c.JSON(http.StatusForbidden, gin.H{"error": "Jadwal kelas belum dimulai"}) return } // ------------------------------------------------------------- // LOGIKA PENOLAKAN TOKEN KADALUARSA // ------------------------------------------------------------- selisihWaktu := jamSelesaiHariIni.Sub(sekarang) sisaMenit := int(selisihWaktu.Minutes()) if sisaMenit <= 0 { fmt.Printf("[VERIFY] Ditolak: Waktu kelas %s sudah habis\n", tokenInput) c.JSON(http.StatusForbidden, gin.H{"error": "Waktu peminjaman sudah habis"}) return } // ------------------------------------------------------------- // JIKA LOLOS SEMUA UJIAN: IZINKAN MASUK & BUKA GEMBOK // ------------------------------------------------------------- DeviceStatusCache["lampu1"] = "on" DeviceStatusCache["lampu2"] = "on" // --- START BLOK MQTT UNLOCK --- broker := os.Getenv("MQTT_BROKER") user := os.Getenv("MQTT_USER") pass := os.Getenv("MQTT_PASSWORD") opts := mqtt.NewClientOptions() opts.AddBroker(broker) opts.SetUsername(user) opts.SetPassword(pass) opts.SetClientID(fmt.Sprintf("Golang-Auth-%d", time.Now().Unix())) client := mqtt.NewClient(opts) if tokenMQTT := client.Connect(); tokenMQTT.Wait() && tokenMQTT.Error() == nil { // 1. Mengirim sinyal otorisasi "UNLOCK" ke Relay dengan retain=true client.Publish("sclass/d101/auth", 0, true, "UNLOCK").Wait() // 2. Opsional: otomatis nyalakan lampu saat token dimasukkan client.Publish("sclass/d101/lampu1", 0, false, "on").Wait() client.Publish("sclass/d101/lampu2", 0, false, "on").Wait() client.Disconnect(250) fmt.Printf("[MQTT] Perintah UNLOCK dan ON berhasil dikirim ke Relay D101\n") } else { fmt.Printf("[MQTT ERROR] Gagal menghubungi broker: %v\n", tokenMQTT.Error()) } // --- END BLOK MQTT UNLOCK --- c.JSON(http.StatusOK, gin.H{ "status": "success", "message": "Token Valid", "duration_minutes": sisaMenit, }) } // ========================================================================= // FUNGSI 2: KONTROL DEVICE VIA HOME ASSISTANT & MQTT // ========================================================================= // ========================================================================= // FUNGSI 2: KONTROL DEVICE VIA HOME ASSISTANT & MQTT // ========================================================================= func ControlHardware(c *gin.Context) { var req DeviceControlRequest if err := c.ShouldBindJSON(&req); err != nil { c.JSON(http.StatusBadRequest, gin.H{"error": "Format data tidak valid"}) return } // 🌟 BLOK 1: KONTROL GEMBOK / OTORISASI RELAY (SANGAT PENTING) if req.Device == "auth" { broker := os.Getenv("MQTT_BROKER") user := os.Getenv("MQTT_USER") pass := os.Getenv("MQTT_PASSWORD") opts := mqtt.NewClientOptions().AddBroker(broker).SetUsername(user).SetPassword(pass).SetClientID(fmt.Sprintf("Golang-Lock-%d", time.Now().Unix())) client := mqtt.NewClient(opts) if token := client.Connect(); token.Wait() && token.Error() == nil { // Retain = true untuk menimpa status UNLOCK lama di server MQTT client.Publish("sclass/d101/auth", 0, true, req.Action).Wait() client.Disconnect(250) fmt.Println("[MQTT] Status Gembok Relay diubah menjadi:", req.Action) } else { fmt.Println("[MQTT ERROR] Gagal terhubung saat mencoba mengunci relay.") } c.JSON(http.StatusOK, gin.H{"status": "success", "message": "Gembok " + req.Action}) return } // --- BLOK 2: KONTROL LAMPU --- if req.Device == "lampu1" || req.Device == "lampu2" { broker := os.Getenv("MQTT_BROKER") user := os.Getenv("MQTT_USER") pass := os.Getenv("MQTT_PASSWORD") opts := mqtt.NewClientOptions() opts.AddBroker(broker) opts.SetUsername(user) opts.SetPassword(pass) opts.SetClientID(fmt.Sprintf("Golang-SCLASS-%d", time.Now().Unix())) client := mqtt.NewClient(opts) if token := client.Connect(); token.Wait() && token.Error() != nil { c.JSON(http.StatusInternalServerError, gin.H{"error": "Golang gagal terhubung ke MQTT Broker"}) return } defer client.Disconnect(250) topic := fmt.Sprintf("sclass/d101/%s", req.Device) token := client.Publish(topic, 0, false, req.Action) token.Wait() DeviceStatusCache[req.Device] = req.Action c.JSON(http.StatusOK, gin.H{"status": "success", "message": fmt.Sprintf("Berhasil %s ke %s", req.Action, req.Device)}) return } // --- BLOK 3: KONTROL AC & PROYEKTOR --- if req.Device == "ac" || req.Device == "projector" { haURL := os.Getenv("HA_URL") haToken := os.Getenv("HA_TOKEN") var entityID string switch req.Device { case "ac": if req.Action == "on" { entityID = "scene.ac_d101_on" } else { entityID = "scene.ac_d101_off" } case "projector": if req.Action == "on" { entityID = "scene.projector_d101_on" } else { entityID = "scene.projector_d101_off" } } apiURL := fmt.Sprintf("%s/api/services/scene/turn_on", haURL) payload := map[string]string{"entity_id": entityID} jsonPayload, _ := json.Marshal(payload) reqHA, err := http.NewRequest("POST", apiURL, bytes.NewBuffer(jsonPayload)) if err == nil { reqHA.Header.Set("Authorization", "Bearer "+haToken) reqHA.Header.Set("Content-Type", "application/json") httpClient := &http.Client{Timeout: 5 * time.Second} resp, _ := httpClient.Do(reqHA) if resp != nil { resp.Body.Close() } } DeviceStatusCache[req.Device] = req.Action c.JSON(http.StatusOK, gin.H{"status": "success", "message": "Scene dipicu: " + entityID}) return } c.JSON(http.StatusBadRequest, gin.H{"error": "Device tidak dikenali"}) } // ========================================================================= // FUNGSI 3: MENDAPATKAN STATUS DAYA DARI HOME ASSISTANT // ========================================================================= func GetPowerStatus(c *gin.Context) { haURL := os.Getenv("HA_URL") haToken := os.Getenv("HA_TOKEN") entityID := "sensor.kwh_meter_power" apiURL := fmt.Sprintf("%s/api/states/%s", haURL, entityID) req, _ := http.NewRequest("GET", apiURL, nil) req.Header.Set("Authorization", "Bearer "+haToken) req.Header.Set("Content-Type", "application/json") client := &http.Client{} resp, err := client.Do(req) if err != nil { c.JSON(http.StatusInternalServerError, gin.H{"power": 0}) return } defer resp.Body.Close() var result map[string]interface{} json.NewDecoder(resp.Body).Decode(&result) powerStr, ok := result["state"].(string) if !ok { c.JSON(http.StatusOK, gin.H{"power": 0}) return } c.JSON(http.StatusOK, gin.H{"power": powerStr}) } // ========================================================================= // FUNGSI 4: MENGIRIM STATUS REAL-TIME KE WEB FRONTEND // ========================================================================= func GetHardwareStatus(c *gin.Context) { c.JSON(http.StatusOK, gin.H{ "status": "success", "data": DeviceStatusCache, }) }