[Valentino Heman Budiarto] ff02b00055 hardware
2026-06-12 12:16:10 +07:00

317 lines
10 KiB
Go

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,
})
}