res.json( status: 'ok', new_happiness_score: kakak.happinessScore, animation: 'happy', reward_points: 1, ); ); CREATE TABLE kakak ( id UUID PRIMARY KEY, name TEXT NOT NULL, happiness_score INT DEFAULT 0, avatar_url TEXT );
// 4️⃣ analytics (pseudo) analytics.track('genjot_success', userId, kakakId ); res.json( status: 'ok'
// 3️⃣ log action await GenjotLog.create( userId, kakakId ); name TEXT NOT NULL
// 2️⃣ update score const kakak = await Kakak.findByPk(kakakId); kakak.happinessScore += 1; await kakak.save(); happiness_score INT DEFAULT 0
return ( <> <button aria-label="Berikan genjotan untuk membuat kakak bahagia" disabled=loading onClick=handleGenjot className="genjot-btn" > loading ? '⏳' : '❤️ Genjot' </button>
const handleGenjot = async () => if (loading) return; setLoading(true); try const res = await postGenjot(kakakId); if (res.status === 'ok') setShowHappy(true); // optional audio const audio = new Audio('/sounds/cheer.mp3'); audio.play(); // hide after 2s setTimeout(() => setShowHappy(false), 2000); finally setLoading(false); ;
// 1️⃣ limit check const recent = await GenjotLog.count( where: userId, kakakId, createdAt: $gt: Date.now() - 60*60*1000 , ); if (recent >= 5) return res.status(429).json( error: 'Rate limit exceeded' );