Skip to content

Latest commit

Β 

History

History
500 lines (402 loc) Β· 17.5 KB

File metadata and controls

500 lines (402 loc) Β· 17.5 KB

Ε½ivotni Ciklus Alarma

Pregled

Alarm prolazi kroz viΕ‘e faza od aktivacije do razreΕ‘enja, sa automatskom eskalacijom ako niko ne reaguje.

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                       ALARM LIFECYCLE                                       β”‚
β”œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€
β”‚                                                                             β”‚
β”‚  AKTIVACIJA ──► ACTIVE ──► ESCALATED_1 ──► ESCALATED_2                      β”‚
β”‚                    β”‚                            β”‚                           β”‚
β”‚                    └─────────► RESPONDING β—„β”€β”€β”€β”€β”€β”˜                           β”‚
β”‚                                    β”‚                                        β”‚
β”‚                                    β–Ό                                        β”‚
β”‚                               ON_SCENE                                      β”‚
β”‚                                    β”‚                                        β”‚
β”‚                                    β–Ό                                        β”‚
β”‚                               RESOLVED                                      β”‚
β”‚                                                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Status alarma

Status Opis
ACTIVE Alarm je aktivan, čeka reakciju respondera na smeni
ESCALATED_1 Niko nije reagovao za 90s, notifikovani svi responderi
ESCALATED_2 Niko nije reagovao za 150s, notifikovani svi članovi
RESPONDING Neko je preuzeo alarm i ide na lokaciju
ON_SCENE Responder je stigao na lokaciju
RESOLVED Situacija je reΕ‘ena
CANCELLED PoΕ‘iljalac je otkazao alarm
FALSE_ALARM Označeno kao lažni alarm

Faza 0: Aktivacija

Korisnik drΕΎi panic button 3 sekunde

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                                     β”‚
β”‚         πŸ”΄                          β”‚
β”‚     DRΕ½I ZA ALARM                   β”‚
β”‚                                     β”‚
β”‚     β–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–ˆβ–‘β–‘β–‘β–‘ 2.1s           β”‚
β”‚                                     β”‚
β”‚     (pusti za otkazivanje)          β”‚
β”‚                                     β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

ZaΕ‘to 3 sekunde?

  • Sprečava slučajne klikove
  • Daje vreme za odustajanje
  • Signal da je akcija namerna

Forma za detalje

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚                                     β”‚
β”‚   πŸ“ Lokacija: Detektovana          β”‚
β”‚                                     β”‚
β”‚   Ε ta se deΕ‘ava? (opciono)          β”‚
β”‚                                     β”‚
β”‚   [Prate me  ] [Tuča     ]          β”‚
β”‚   [Kradu     ] [Preti mi ]          β”‚
β”‚                                     β”‚
β”‚   [ Drugo: __________________ ]     β”‚
β”‚                                     β”‚
β”‚   [🚨 POΕ ALJI ALARM]                β”‚
β”‚                                     β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Faza 1: ACTIVE (T=0)

Backend logika

async function createAlarm({ location, message, group_id }) {
  // 1. Kreiraj alarm
  const alarm = await db.insert("alarms", {
    group_id,
    triggered_by: user._id,
    location_lat: location.lat,
    location_lng: location.lng,
    message,
    status: "ACTIVE",
    created_at: Date.now(),
  });

  // 2. NaΔ‘i RESPONDERE na smeni
  const activeResponders = await getActiveResponders(group_id);

  // 3. Sortiraj po blizini
  const sorted = sortByDistance(activeResponders, location);

  // 4. Kreiraj alarm_responses za svakog
  for (const responder of sorted) {
    await db.insert("alarm_responses", {
      alarm_id: alarm._id,
      user_id: responder.user_id,
      notified_at: Date.now(),
      distance_meters: responder.distance,
    });
  }

  // 5. PoΕ‘alji Telegram
  await sendTelegramAlert(group_id, alarm);

  // 6. ZakaΕΎi eskalaciju
  await scheduler.runAfter(90_000, "escalateAlarm", { alarm_id: alarm._id });

  return alarm;
}

Ε ta vide responderi na smeni

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  🚨 ALARM - OΕ  KovačiΔ‡                      β”‚
β”‚                                             β”‚
β”‚  πŸ“ 300m od tebe (Kod fontane)              β”‚
β”‚  πŸ’¬ "Prate me"                              β”‚
β”‚  πŸ‘€ Od: Dete MarkoviΔ‡a                      β”‚
β”‚                                             β”‚
β”‚  Ti si 1 od 2 dostupna respondera           β”‚
β”‚                                             β”‚
β”‚  ⏱️ Eskalacija za: 1:23                     β”‚
β”‚                                             β”‚
β”‚  [πŸƒ PREUZIMAM]                             β”‚
β”‚                                             β”‚
β”‚  Ne mogu jer: [ Izaberi razlog β–Ό ]          β”‚
β”‚                                             β”‚
β”‚  πŸ‘οΈ Videlo: 2    βœ… Preuzelo: 0             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Telegram poruka

🚨 *ALARM*

πŸ“ Lokacija: https://maps.google.com/?q=44.81,20.46
πŸ’¬ "Prate me"
πŸ‘€ Od: Dete MarkoviΔ‡a

πŸ‘† Otvori app: https://patrola.rs/alarm/xyz

Faza 2: Eskalacija 1 (T=90s)

Trigger

Scheduled function se pokreće 90 sekundi nakon kreiranja alarma.

async function escalateAlarm({ alarm_id }) {
  const alarm = await db.get(alarm_id);

  // Ako je već neko preuzeo, ne radi niőta
  if (alarm.status !== "ACTIVE") return;

  // Update status
  await db.patch(alarm_id, {
    status: "ESCALATED_1",
    escalated_1_at: Date.now(),
  });

  // Notifikuj SVE respondere (ne samo one na smeni)
  const allResponders = await getAllResponders(alarm.group_id);
  await sendUrgentNotification(allResponders, alarm, "ESCALATED");

  // Zakaži sledeću eskalaciju
  await scheduler.runAfter(60_000, "escalateAlarmFinal", { alarm_id });
}

Poruka za sve respondere

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  ⚠️ ESKALIRAN ALARM - OΕ  KovačiΔ‡           β”‚
β”‚                                             β”‚
β”‚  Niko od dostupnih respondera nije          β”‚
│  preuzeo alarm već 90 sekundi!              │
β”‚                                             β”‚
β”‚  πŸ“ Kod fontane                             β”‚
β”‚  πŸ’¬ "Prate me"                              β”‚
β”‚                                             β”‚
β”‚  [πŸƒ PREUZIMAM]                             β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Faza 3: Eskalacija 2 - Broadcast (T=150s)

Trigger

Ako niko nije reagovao ni nakon eskalacije 1.

async function escalateAlarmFinal({ alarm_id }) {
  const alarm = await db.get(alarm_id);

  if (alarm.status !== "ESCALATED_1") return;

  await db.patch(alarm_id, {
    status: "ESCALATED_2",
    escalated_2_at: Date.now(),
  });

  // Notifikuj SVE članove grupe (uključujuΔ‡i RODITELJE)
  const allMembers = await getAllMembers(alarm.group_id);
  await sendCriticalNotification(allMembers, alarm);

  // Pozovi admina direktno (opciono)
  await callAdmin(alarm.group_id, alarm);
}

Poruka za sve članove

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  πŸ†˜ KRITIČNO - ALARM BEZ ODGOVORA           β”‚
β”‚                                             β”‚
│  Alarm je aktivan već 2.5 minuta            │
β”‚  i NIKO nije preuzeo!                       β”‚
β”‚                                             β”‚
β”‚  Ako ste u blizini Ε‘kole KovačiΔ‡,           β”‚
β”‚  molimo reagujte!                           β”‚
β”‚                                             β”‚
β”‚  [OTVORI DETALJE]                           β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Faza 4: RESPONDING (neko preuzeo)

Kada responder klikne "Preuzimam"

async function takeAlarm({ alarm_id, taken_by }) {
  const alarm = await db.get(alarm_id);

  if (!["ACTIVE", "ESCALATED_1", "ESCALATED_2"].includes(alarm.status)) {
    throw new Error("Alarm nije u stanju za preuzimanje");
  }

  // Update alarm
  await db.patch(alarm_id, {
    status: "RESPONDING",
    responded_at: Date.now(),
  });

  // Update alarm_response
  await db.patch(responseId, {
    response: "ACCEPTED",
    response_at: Date.now(),
    eta_minutes: calculateETA(responderLocation, alarmLocation),
  });

  // OTKAΕ½I zakazanu eskalaciju
  await scheduler.cancel(escalationJobId);

  // Telegram update
  await sendTelegramUpdate(`βœ… ${taken_by} je preuzeo alarm`);
}

Ε ta vidi responder

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  βœ… Preuzeo si alarm                         β”‚
β”‚                                             β”‚
β”‚  πŸ“ Navigacija do lokacije:                 β”‚
β”‚  [OTVORI GOOGLE MAPS]                       β”‚
β”‚                                             β”‚
β”‚  πŸ“ž Pozovi dete: +381 63 xxx xxxx           β”‚
β”‚  πŸ“ž Pozovi roditelja: +381 64 xxx xxxx      β”‚
β”‚                                             β”‚
β”‚  Javi kad stigneΕ‘:                          β”‚
β”‚  [βœ… STIGAO SAM]                            β”‚
β”‚                                             β”‚
β”‚  Problem?                                   β”‚
β”‚  [❌ MORAM DA ODUSTANEM]                    β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Ε ta vide ostali

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  🚨 ALARM                                   β”‚
β”‚                                             β”‚
β”‚  βœ… Petar je preuzeo                        β”‚
β”‚  πŸ“ Udaljen ~400m                           β”‚
β”‚  ⏱️ ETA: ~4 min                             β”‚
β”‚                                             β”‚
β”‚  [πŸ“ VIDI LOKACIJU]                         β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Ε ta vidi poΕ‘iljalac (dete/roditelj)

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  βœ… PomoΔ‡ je na putu!                        β”‚
β”‚                                             β”‚
β”‚  πŸ‘€ Petar P. dolazi                         β”‚
β”‚  πŸ“ Udaljen ~400m                           β”‚
β”‚  ⏱️ Očekivano vreme: ~4 min                 β”‚
β”‚                                             β”‚
β”‚  Ako je laΕΎna uzbuna:                       β”‚
β”‚  [OTKAΕ½I ALARM]                             β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Faza 5: ON_SCENE (stigao)

Kada responder klikne "Stigao sam"

async function arriveAtScene({ alarm_id }) {
  await db.patch(alarm_id, {
    status: "ON_SCENE",
  });

  await db.patch(responseId, {
    arrived_at: Date.now(),
  });

  await sendTelegramUpdate("πŸ“ Responder je stigao na lokaciju");
}

Interface za razreΕ‘enje

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  πŸ“ Na licu mesta                           β”‚
β”‚                                             β”‚
β”‚  Kad se situacija reΕ‘i, označi:             β”‚
β”‚                                             β”‚
β”‚  [βœ… REΕ ENO - Sve OK]                       β”‚
β”‚                                             β”‚
β”‚  [⚠️ REΕ ENO - Potrebna intervencija]        β”‚
β”‚     (treba pozvati policiju, roditelje...)  β”‚
β”‚                                             β”‚
β”‚  [❌ LAΕ½NI ALARM]                           β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Faza 6: RESOLVED (razreΕ‘eno)

Forma za razreΕ‘enje

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  Kako je reΕ‘eno?                            β”‚
β”‚                                             β”‚
β”‚  [β—‹] Situacija procenjena kao neaktuelna    β”‚
β”‚  [β—‹] Dete sprovedeno do bezbedne lokacije   β”‚
β”‚  [β—‹] Pozvana policija                       β”‚
β”‚  [β—‹] Pozvani roditelji                       β”‚
β”‚  [β—‹] Drugo                                  β”‚
β”‚                                             β”‚
β”‚  Dodatne napomene:                          β”‚
β”‚  [ ___________________________________ ]    β”‚
β”‚                                             β”‚
β”‚  [ZAVRΕ I]                                   β”‚
β”‚                                             β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜

Backend

async function resolveAlarm({ alarm_id, resolution, notes }) {
  await db.patch(alarm_id, {
    status: "RESOLVED",
    resolved_at: Date.now(),
    resolution_notes: notes,
    resolved_by: user._id,
  });

  // Update responder statistike
  await updateResponderStats(responderId, {
    alarmsResolved: increment(1),
    // ...
  });

  // Telegram
  await sendTelegramUpdate("βœ… Alarm razreΕ‘en");

  // Audit log
  await db.insert("audit_log", {
    action: "ALARM_RESOLVED",
    alarm_id,
    user_id: user._id,
    details: JSON.stringify({ resolution, notes }),
  });
}

Dijagram state machine

                              CANCELLED
                                 β–²
                                 β”‚
        β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€
        β”‚                        β”‚
        β”‚                        β”‚
        β–Ό                        β”‚
     ACTIVE ──────90s────► ESCALATED_1 ──────60s────► ESCALATED_2
        β”‚                        β”‚                         β”‚
        β”‚                        β”‚                         β”‚
        β”‚    β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”΄β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
        β”‚    β”‚
        β”‚    β”‚ neko preuzme
        β–Ό    β–Ό
     RESPONDING
        β”‚
        β”‚ stigao
        β–Ό
     ON_SCENE
        β”‚
        β”‚ reΕ‘i
        β–Ό
     RESOLVED ◄──────────────────────────────────────── FALSE_ALARM

Vremenska linija tipičnog alarma

T=0      Alarm kreiran, status: ACTIVE
         β†’ Notifikovani responderi na smeni (2)

T=15s    Responder 1 video alarm
T=32s    Responder 1 kliknuo "Ne mogu" (razlog: na poslu)

T=45s    Responder 2 video alarm
T=52s    Responder 2 kliknuo "PREUZIMAM"
         β†’ Status: RESPONDING
         β†’ Eskalacija otkazana

T=4min   Responder 2 kliknuo "STIGAO SAM"
         β†’ Status: ON_SCENE

T=9min   Responder 2 označio kao RESOLVED
         β†’ Status: RESOLVED
         β†’ Ukupno vreme: 9 minuta

Edge case-ovi

Situacija PonaΕ‘anje
Responder preuzme pa odustane Alarm se vraΔ‡a u ACTIVE, eskalacija kreΔ‡e ispočetka
Dva respondera preuzmu istovremeno Oba se beleΕΎe kao ACCEPTED, koordinacija u app-u
Poőiljalac otkaže dok neko ide Responder dobija notifikaciju "Alarm povučen"
Niko ne reaguje ni na broadcast Ostaje u ESCALATED_2, admin notifikovan

Dokument kreiran: Januar 2026