SupaBase entry done
This commit is contained in:
		
							parent
							
								
									8db95e6d40
								
							
						
					
					
						commit
						ec8726cf94
					
				| @ -22176,7 +22176,7 @@ GameObject: | |||||||
|   m_Icon: {fileID: 0} |   m_Icon: {fileID: 0} | ||||||
|   m_NavMeshLayer: 0 |   m_NavMeshLayer: 0 | ||||||
|   m_StaticEditorFlags: 0 |   m_StaticEditorFlags: 0 | ||||||
|   m_IsActive: 1 |   m_IsActive: 0 | ||||||
| --- !u!114 &292487262 | --- !u!114 &292487262 | ||||||
| MonoBehaviour: | MonoBehaviour: | ||||||
|   m_ObjectHideFlags: 0 |   m_ObjectHideFlags: 0 | ||||||
|  | |||||||
| @ -47,14 +47,14 @@ public class BodyLinkHandler : MonoBehaviour, IPointerClickHandler, IPointerExit | |||||||
|                 EmailData email = emailPanel.Email; |                 EmailData email = emailPanel.Email; | ||||||
|                 if (email != null) |                 if (email != null) | ||||||
|                 { |                 { | ||||||
|                     SceneOutcomeManager.Instance?.Clicked(email); |  | ||||||
|                     //UserActionLogger.Instance?.Log($"Clicked link '{linkID}' in email from '{email.senderName}'"); |                     //UserActionLogger.Instance?.Log($"Clicked link '{linkID}' in email from '{email.senderName}'"); | ||||||
|                     string englishLog = $"Clicked link '{linkID}' in email from '{email.senderName}'"; |                     string englishLog = $"Clicked link '{linkID}' in email from '{email.senderName}'"; | ||||||
|                     string arabicLog = $"تم الضغط على الرابط '{linkID}' في البريد من '{email.senderName}'"; |                     string arabicLog = $"تم الضغط على الرابط '{linkID}' في البريد من '{email.senderName}'"; | ||||||
| 
 |  | ||||||
|                     UserActionLogger.Instance?.Log(englishLog, arabicLog); |                     UserActionLogger.Instance?.Log(englishLog, arabicLog); | ||||||
|  | 
 | ||||||
|                     bool isOptimal = !email.isPhishing; |                     bool isOptimal = !email.isPhishing; | ||||||
|                     SupabaseEventLogger.Instance?.LogDecisionEvent(isOptimal); |                     SupabaseEventLogger.Instance?.LogDecisionEvent(isOptimal); | ||||||
|  |                     SceneOutcomeManager.Instance?.Clicked(email); | ||||||
| 
 | 
 | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|  | |||||||
| @ -65,21 +65,42 @@ public class UserActionLogger : MonoBehaviour | |||||||
|     [ContextMenu("ShowSummary")] |     [ContextMenu("ShowSummary")] | ||||||
|     public void ShowSummary() |     public void ShowSummary() | ||||||
|     { |     { | ||||||
|         if (summaryText != null) |         if (summaryText == null) | ||||||
|         { |             return; | ||||||
| 
 | 
 | ||||||
|             bool isArabic = LanguageManager.Instance != null && |         string logs = GetFullLog(); | ||||||
|                             LanguageManager.Instance.currentLanguage == "Arabic"; | 
 | ||||||
|             string logs = GetFullLog(); |         // Default to showing original logs | ||||||
|             if (isArabic) |         summaryText.text = logs; | ||||||
|             { | 
 | ||||||
|                 summaryText.text = ArabicFixer.Fix(logs); |         // If Arabic language is active, fix and apply Arabic font | ||||||
|                 summaryText.font = LanguageManager.Instance.fontArabic; |         if (LanguageManager.Instance != null && | ||||||
|                 summaryText.ForceMeshUpdate(); |             LanguageManager.Instance.currentLanguage == "Arabic") | ||||||
|             } |         { | ||||||
|  |             summaryText.text = ArabicFixer.Fix(logs); | ||||||
|  |             summaryText.font = LanguageManager.Instance.fontArabic; | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         summaryText.ForceMeshUpdate(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     //public void ShowSummary() | ||||||
|  |     //{ | ||||||
|  |     //    if (summaryText != null) | ||||||
|  |     //    { | ||||||
|  | 
 | ||||||
|  |     //        bool isArabic = LanguageManager.Instance != null && | ||||||
|  |     //                        LanguageManager.Instance.currentLanguage == "Arabic"; | ||||||
|  |     //        string logs = GetFullLog(); | ||||||
|  |     //        if (isArabic) | ||||||
|  |     //        { | ||||||
|  |     //            summaryText.text = ArabicFixer.Fix(logs); | ||||||
|  |     //            summaryText.font = LanguageManager.Instance.fontArabic; | ||||||
|  |     //            summaryText.ForceMeshUpdate(); | ||||||
|  |     //        } | ||||||
|  |     //    } | ||||||
|  |     //} | ||||||
|  | 
 | ||||||
|     public void ClearLog() |     public void ClearLog() | ||||||
|     { |     { | ||||||
|         logs.Clear(); |         logs.Clear(); | ||||||
|  | |||||||
| @ -1,15 +1,17 @@ | |||||||
| using UnityEngine; | using UnityEngine; | ||||||
|  | using UnityEngine.Networking; | ||||||
| using System; | using System; | ||||||
|  | using System.Collections; | ||||||
| using System.Collections.Generic; | using System.Collections.Generic; | ||||||
| using System.Threading.Tasks; |  | ||||||
| using Supabase; // Make sure SupabaseManager initializes this correctly |  | ||||||
| using Postgrest.Models; |  | ||||||
| using Postgrest.Attributes; |  | ||||||
| 
 | 
 | ||||||
| public class SupabaseEventLogger : MonoBehaviour | public class SupabaseEventLogger : MonoBehaviour | ||||||
| { | { | ||||||
|     public static SupabaseEventLogger Instance; |     public static SupabaseEventLogger Instance; | ||||||
| 
 | 
 | ||||||
|  |     [Header("Supabase")] | ||||||
|  |     public string supabaseUrl = "https://vihjspljbslozbjzxutl.supabase.co"; | ||||||
|  |     public string supabaseAnonKey = "YOUR_ANON_KEY_HERE"; | ||||||
|  | 
 | ||||||
|     private DateTime sessionStartTime; |     private DateTime sessionStartTime; | ||||||
| 
 | 
 | ||||||
|     private void Awake() |     private void Awake() | ||||||
| @ -20,86 +22,29 @@ public class SupabaseEventLogger : MonoBehaviour | |||||||
|             Destroy(gameObject); |             Destroy(gameObject); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     [Serializable] | ||||||
|     /// Call this at the start of the game session. |     public class GameEventPayload | ||||||
|     /// </summary> |  | ||||||
|     public async void StartSession() |  | ||||||
|     { |     { | ||||||
|         sessionStartTime = DateTime.UtcNow; |         public string event_key; | ||||||
| 
 |         public string timestamp; | ||||||
|         var gameEvent = new GameEvent |         public string user_id; | ||||||
|         { |  | ||||||
|             Id = Guid.NewGuid(), // <== Ensure this is explicitly set |  | ||||||
|             EventKey = "game_session_started", |  | ||||||
|             Timestamp = sessionStartTime, |  | ||||||
|             UserId = "user123" |  | ||||||
|         }; |  | ||||||
|         await Client.Instance.From<GameEvent>().Insert(gameEvent); |  | ||||||
|         Debug.Log("✅ Supabase Event: game_session_started"); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     [Serializable] | ||||||
|     /// Logs optimal/suboptimal decisions at runtime. |     public class GameAttemptPayload | ||||||
|     /// </summary> |  | ||||||
|     public async void LogDecisionEvent(bool isOptimal) |  | ||||||
|     { |     { | ||||||
|         string eventKey = isOptimal ? "game_optimal_decision_made" : "game_suboptimal_decision_made"; |         public string game_id; | ||||||
| 
 |         public string scenario_id; | ||||||
|         var gameEvent = new GameEvent |         public string user_id; | ||||||
|         { |         public int attempt_number; | ||||||
|             Id = Guid.NewGuid(), |         public string start_timestamp; | ||||||
|             EventKey = eventKey, |         public string end_timestamp; | ||||||
|             Timestamp = DateTime.UtcNow, |         public int duration_seconds; | ||||||
|             UserId = "user123" |         public float final_score_percentage; | ||||||
|         }; |         public bool pass_fail_status; | ||||||
| 
 |         public int optimal_decisions_count; | ||||||
|         await Client.Instance.From<GameEvent>().Insert(gameEvent); |         public int suboptimal_decisions_count; | ||||||
|         Debug.Log($"✅ Supabase Event: {eventKey}"); |         public string key_decisions_log; | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// Completes the session and submits full results to phishing_game_attempts table. |  | ||||||
|     /// </summary> |  | ||||||
|     public async void CompleteSessionAndSubmitResult(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog = null) |  | ||||||
|     { |  | ||||||
|         var endTime = DateTime.UtcNow; |  | ||||||
|         int duration = (int)(endTime - sessionStartTime).TotalSeconds; |  | ||||||
| 
 |  | ||||||
|         // Log completion events |  | ||||||
|         await Client.Instance.From<GameEvent>().Insert(new GameEvent |  | ||||||
|         { |  | ||||||
|             Id = Guid.NewGuid(), |  | ||||||
|             EventKey = "game_session_completed", |  | ||||||
|             Timestamp = endTime, |  | ||||||
|             UserId = userId |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         await Client.Instance.From<GameEvent>().Insert(new GameEvent |  | ||||||
|         {Id = Guid.NewGuid(), |  | ||||||
|             EventKey = "game_score_recorded", |  | ||||||
|             Timestamp = endTime, |  | ||||||
|             UserId = userId |  | ||||||
|         }); |  | ||||||
| 
 |  | ||||||
|         // Insert session result |  | ||||||
|         var gameAttempt = new GameAttempt |  | ||||||
|         { |  | ||||||
|             GameId = "phishing-awareness-1", |  | ||||||
|             ScenarioId = scenarioId, |  | ||||||
|             UserId = userId, |  | ||||||
|             AttemptNumber = 1, |  | ||||||
|             StartTime = sessionStartTime, |  | ||||||
|             EndTime = endTime, |  | ||||||
|             DurationSeconds = duration, |  | ||||||
|             Score = passed ? 100 : 50, |  | ||||||
|             Passed = passed, |  | ||||||
|             Optimal = optimal, |  | ||||||
|             Suboptimal = suboptimal, |  | ||||||
|             KeyDecisionsLogJson = decisionLog != null ? JsonUtility.ToJson(new DecisionLogWrapper { decisions = decisionLog }) : "[]" |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         await Client.Instance.From<GameAttempt>().Insert(gameAttempt); |  | ||||||
|         Debug.Log("✅ Supabase Game Result Submitted"); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Serializable] |     [Serializable] | ||||||
| @ -115,66 +60,501 @@ public class SupabaseEventLogger : MonoBehaviour | |||||||
|     { |     { | ||||||
|         public List<Decision> decisions; |         public List<Decision> decisions; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public void StartSession() | ||||||
|  |     { | ||||||
|  |         sessionStartTime = DateTime.UtcNow; | ||||||
|  | 
 | ||||||
|  |         GameEventPayload payload = new GameEventPayload | ||||||
|  |         { | ||||||
|  |             event_key = "game_session_started", | ||||||
|  |             timestamp = sessionStartTime.ToString("o"), | ||||||
|  |             user_id = "user123" | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         StartCoroutine(PostToSupabase("game_events", JsonUtility.ToJson(payload))); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void LogDecisionEvent(bool isOptimal) | ||||||
|  |     { | ||||||
|  |         string eventKey = isOptimal ? "game_optimal_decision_made" : "game_suboptimal_decision_made"; | ||||||
|  | 
 | ||||||
|  |         GameEventPayload payload = new GameEventPayload | ||||||
|  |         { | ||||||
|  |             event_key = eventKey, | ||||||
|  |             timestamp = DateTime.UtcNow.ToString("o"), | ||||||
|  |             user_id = "user123" | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         StartCoroutine(PostToSupabase("game_events", JsonUtility.ToJson(payload))); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void CompleteSessionAndSubmitResult(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog = null) | ||||||
|  |     { | ||||||
|  |         var endTime = DateTime.UtcNow; | ||||||
|  |         int duration = (int)(endTime - sessionStartTime).TotalSeconds; | ||||||
|  | 
 | ||||||
|  |         // Submit game_session_completed event | ||||||
|  |         GameEventPayload completedEvent = new GameEventPayload | ||||||
|  |         { | ||||||
|  |             event_key = "game_session_completed", | ||||||
|  |             timestamp = endTime.ToString("o"), | ||||||
|  |             user_id = userId | ||||||
|  |         }; | ||||||
|  |         StartCoroutine(PostToSupabase("game_events", JsonUtility.ToJson(completedEvent))); | ||||||
|  | 
 | ||||||
|  |         // Submit game_score_recorded event | ||||||
|  |         GameEventPayload scoreEvent = new GameEventPayload | ||||||
|  |         { | ||||||
|  |             event_key = "game_score_recorded", | ||||||
|  |             timestamp = endTime.ToString("o"), | ||||||
|  |             user_id = userId | ||||||
|  |         }; | ||||||
|  |         StartCoroutine(PostToSupabase("game_events", JsonUtility.ToJson(scoreEvent))); | ||||||
|  | 
 | ||||||
|  |         // Submit final game attempt data | ||||||
|  |         GameAttemptPayload attempt = new GameAttemptPayload | ||||||
|  |         { | ||||||
|  |             game_id = "phishing-awareness-1", | ||||||
|  |             scenario_id = scenarioId, | ||||||
|  |             user_id = userId, | ||||||
|  |             attempt_number = 1, | ||||||
|  |             start_timestamp = sessionStartTime.ToString("o"), | ||||||
|  |             end_timestamp = endTime.ToString("o"), | ||||||
|  |             duration_seconds = duration, | ||||||
|  |             final_score_percentage = passed ? 100 : 50, | ||||||
|  |             pass_fail_status = passed, | ||||||
|  |             optimal_decisions_count = optimal, | ||||||
|  |             suboptimal_decisions_count = suboptimal, | ||||||
|  |             key_decisions_log = decisionLog != null ? JsonUtility.ToJson(new DecisionLogWrapper { decisions = decisionLog }) : "[]" | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         StartCoroutine(PostToSupabase("phishing_game_attempts", JsonUtility.ToJson(attempt))); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private IEnumerator PostToSupabase(string table, string jsonBody) | ||||||
|  |     { | ||||||
|  |         string url = $"{supabaseUrl}/rest/v1/{table}"; | ||||||
|  |         UnityWebRequest request = new UnityWebRequest(url, "POST"); | ||||||
|  |         byte[] bodyRaw = System.Text.Encoding.UTF8.GetBytes(jsonBody); | ||||||
|  |         request.uploadHandler = new UploadHandlerRaw(bodyRaw); | ||||||
|  |         request.downloadHandler = new DownloadHandlerBuffer(); | ||||||
|  | 
 | ||||||
|  |         request.SetRequestHeader("Content-Type", "application/json"); | ||||||
|  |         request.SetRequestHeader("apikey", supabaseAnonKey); | ||||||
|  |         request.SetRequestHeader("Authorization", "Bearer " + supabaseAnonKey); | ||||||
|  |         request.SetRequestHeader("Prefer", "return=representation"); | ||||||
|  | 
 | ||||||
|  |         yield return request.SendWebRequest(); | ||||||
|  | 
 | ||||||
|  |         if (request.result == UnityWebRequest.Result.Success) | ||||||
|  |         { | ||||||
|  |             Debug.Log($"✅ Supabase POST to {table}: " + request.downloadHandler.text); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             Debug.LogError($"❌ Supabase POST Failed ({table}): {request.responseCode}\n{request.error}\n{request.downloadHandler.text}"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Existing SupabaseEventLogger class here... |  | ||||||
| // ↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓ |  | ||||||
| 
 | 
 | ||||||
| [Table("game_events")] |  | ||||||
| public class GameEvent : BaseModel |  | ||||||
| { |  | ||||||
|     [PrimaryKey("id", false)] |  | ||||||
|     public Guid Id { get; set; } |  | ||||||
| 
 | 
 | ||||||
|     [Column("event_key")] | //using UnityEngine; | ||||||
|     public string EventKey { get; set; } | //using System; | ||||||
|  | //using System.Collections; | ||||||
|  | //using System.Collections.Generic; | ||||||
|  | //using System.Threading.Tasks; | ||||||
|  | //using Supabase; | ||||||
|  | //using Postgrest.Models; | ||||||
|  | //using Postgrest.Attributes; | ||||||
|  | //using static SupabaseTestInsert; | ||||||
| 
 | 
 | ||||||
|     [Column("timestamp")] | //[Table("game_events")] | ||||||
|     public DateTime Timestamp { get; set; } | //public class GameEvent : BaseModel | ||||||
|  | //{ | ||||||
|  | //    [PrimaryKey("id", false)] | ||||||
|  | //    public Guid Id { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("user_id")] | //    [Column("event_key")] | ||||||
|     public string UserId { get; set; } | //    public string EventKey { get; set; } | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| [Table("phishing_game_attempts")] | //    [Column("timestamp")] | ||||||
| public class GameAttempt : BaseModel | //    public DateTime Timestamp { get; set; } | ||||||
| { |  | ||||||
|     [PrimaryKey("id", false)] |  | ||||||
|     public Guid Id { get; set; } = Guid.NewGuid(); |  | ||||||
| 
 | 
 | ||||||
|     [Column("game_id")] | //    [Column("user_id")] | ||||||
|     public string GameId { get; set; } | //    public string UserId { get; set; } | ||||||
|  | //} | ||||||
| 
 | 
 | ||||||
|     [Column("scenario_id")] | //[Table("phishing_game_attempts")] | ||||||
|     public string ScenarioId { get; set; } | //public class GameAttempt : BaseModel | ||||||
|  | //{ | ||||||
|  | //    [PrimaryKey("id", false)] | ||||||
|  | //    public Guid Id { get; set; } = Guid.NewGuid(); | ||||||
| 
 | 
 | ||||||
|     [Column("user_id")] | //    [Column("game_id")] | ||||||
|     public string UserId { get; set; } | //    public string GameId { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("attempt_number")] | //    [Column("scenario_id")] | ||||||
|     public int AttemptNumber { get; set; } | //    public string ScenarioId { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("start_timestamp")] | //    [Column("user_id")] | ||||||
|     public DateTime StartTime { get; set; } | //    public string UserId { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("end_timestamp")] | //    [Column("attempt_number")] | ||||||
|     public DateTime EndTime { get; set; } | //    public int AttemptNumber { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("duration_seconds")] | //    [Column("start_timestamp")] | ||||||
|     public int DurationSeconds { get; set; } | //    public DateTime StartTime { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("final_score_percentage")] | //    [Column("end_timestamp")] | ||||||
|     public float Score { get; set; } | //    public DateTime EndTime { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("pass_fail_status")] | //    [Column("duration_seconds")] | ||||||
|     public bool Passed { get; set; } | //    public int DurationSeconds { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("optimal_decisions_count")] | //    [Column("final_score_percentage")] | ||||||
|     public int Optimal { get; set; } | //    public float Score { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("suboptimal_decisions_count")] | //    [Column("pass_fail_status")] | ||||||
|     public int Suboptimal { get; set; } | //    public bool Passed { get; set; } | ||||||
| 
 | 
 | ||||||
|     [Column("key_decisions_log")] | //    [Column("optimal_decisions_count")] | ||||||
|     public string KeyDecisionsLogJson { get; set; } | //    public int Optimal { get; set; } | ||||||
| } | 
 | ||||||
|  | //    [Column("suboptimal_decisions_count")] | ||||||
|  | //    public int Suboptimal { get; set; } | ||||||
|  | 
 | ||||||
|  | //    [Column("key_decisions_log")] | ||||||
|  | //    public string KeyDecisionsLogJson { get; set; } | ||||||
|  | //} | ||||||
|  | 
 | ||||||
|  | //public class SupabaseEventLogger : MonoBehaviour | ||||||
|  | //{ | ||||||
|  | //    public static SupabaseEventLogger Instance; | ||||||
|  | 
 | ||||||
|  | //    private DateTime sessionStartTime; | ||||||
|  | 
 | ||||||
|  | //    private void Awake() | ||||||
|  | //    { | ||||||
|  | //        if (Instance == null) | ||||||
|  | //            Instance = this; | ||||||
|  | //        else | ||||||
|  | //            Destroy(gameObject); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    public void StartSession() | ||||||
|  | //    { | ||||||
|  | //        StartCoroutine(StartSessionCoroutine()); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private IEnumerator StartSessionCoroutine() | ||||||
|  | //    { | ||||||
|  | //        var task = StartSessionAsync(); | ||||||
|  | //        while (!task.IsCompleted) | ||||||
|  | //            yield return null; | ||||||
|  | 
 | ||||||
|  | //        if (task.Exception != null) | ||||||
|  | //            Debug.LogError("❌ Supabase Error: " + task.Exception.InnerException?.Message); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private async Task StartSessionAsync() | ||||||
|  | //    { | ||||||
|  | //        sessionStartTime = DateTime.UtcNow; | ||||||
|  | 
 | ||||||
|  | //        var gameEvent = new GameEvent | ||||||
|  | //        { | ||||||
|  | //            Id = Guid.NewGuid(), | ||||||
|  | //            EventKey = "game_session_started", | ||||||
|  | //            Timestamp = sessionStartTime, | ||||||
|  | //            UserId = "user123" | ||||||
|  | //        }; | ||||||
|  | 
 | ||||||
|  | //        await Client.Instance.From<GameEvent>().Insert(gameEvent); | ||||||
|  | //        Debug.Log("✅ Supabase Event: game_session_started"); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    public void LogDecisionEvent(bool isOptimal) | ||||||
|  | //    { | ||||||
|  | //        StartCoroutine(LogDecisionCoroutine(isOptimal)); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private IEnumerator LogDecisionCoroutine(bool isOptimal) | ||||||
|  | //    { | ||||||
|  | //        var task = LogDecisionAsync(isOptimal); | ||||||
|  | //        while (!task.IsCompleted) | ||||||
|  | //            yield return null; | ||||||
|  | 
 | ||||||
|  | //        if (task.Exception != null) | ||||||
|  | //            Debug.LogError("❌ Supabase Error: " + task.Exception.InnerException?.Message); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private async Task LogDecisionAsync(bool isOptimal) | ||||||
|  | //    { | ||||||
|  | //        string eventKey = isOptimal ? "game_optimal_decision_made" : "game_suboptimal_decision_made"; | ||||||
|  | 
 | ||||||
|  | //        var gameEvent = new GameEvent | ||||||
|  | //        { | ||||||
|  | //            Id = Guid.NewGuid(), | ||||||
|  | //            EventKey = eventKey, | ||||||
|  | //            Timestamp = DateTime.UtcNow, | ||||||
|  | //            UserId = "user123" | ||||||
|  | //        }; | ||||||
|  | 
 | ||||||
|  | //        await Client.Instance.From<GameEvent>().Insert(gameEvent); | ||||||
|  | //        Debug.Log($"✅ Supabase Event: {eventKey}"); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    public void CompleteSessionAndSubmitResult(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog = null) | ||||||
|  | //    { | ||||||
|  | //        StartCoroutine(CompleteSessionCoroutine(userId, passed, optimal, suboptimal, scenarioId, decisionLog)); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private IEnumerator CompleteSessionCoroutine(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog) | ||||||
|  | //    { | ||||||
|  | //        var task = CompleteSessionAsync(userId, passed, optimal, suboptimal, scenarioId, decisionLog); | ||||||
|  | //        while (!task.IsCompleted) | ||||||
|  | //            yield return null; | ||||||
|  | 
 | ||||||
|  | //        if (task.Exception != null) | ||||||
|  | //            Debug.LogError("❌ Supabase Error: " + task.Exception.InnerException?.Message); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    private async Task CompleteSessionAsync(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog) | ||||||
|  | //    { | ||||||
|  | //        var endTime = DateTime.UtcNow; | ||||||
|  | //        int duration = (int)(endTime - sessionStartTime).TotalSeconds; | ||||||
|  | 
 | ||||||
|  | //        await Client.Instance.From<GameEvent>().Insert(new GameEvent | ||||||
|  | //        { | ||||||
|  | //            Id = Guid.NewGuid(), | ||||||
|  | //            EventKey = "game_session_completed", | ||||||
|  | //            Timestamp = endTime, | ||||||
|  | //            UserId = userId | ||||||
|  | //        }); | ||||||
|  | 
 | ||||||
|  | //        await Client.Instance.From<GameEvent>().Insert(new GameEvent | ||||||
|  | //        { | ||||||
|  | //            Id = Guid.NewGuid(), | ||||||
|  | //            EventKey = "game_score_recorded", | ||||||
|  | //            Timestamp = endTime, | ||||||
|  | //            UserId = userId | ||||||
|  | //        }); | ||||||
|  | 
 | ||||||
|  | //        var gameAttempt = new GameAttempt | ||||||
|  | //        { | ||||||
|  | //            Id = Guid.NewGuid(), | ||||||
|  | //            GameId = "phishing-awareness-1", | ||||||
|  | //            ScenarioId = scenarioId, | ||||||
|  | //            UserId = userId, | ||||||
|  | //            AttemptNumber = 1, | ||||||
|  | //            StartTime = sessionStartTime, | ||||||
|  | //            EndTime = endTime, | ||||||
|  | //            DurationSeconds = duration, | ||||||
|  | //            Score = passed ? 100 : 50, | ||||||
|  | //            Passed = passed, | ||||||
|  | //            Optimal = optimal, | ||||||
|  | //            Suboptimal = suboptimal, | ||||||
|  | //            KeyDecisionsLogJson = decisionLog != null ? JsonUtility.ToJson(new DecisionLogWrapper { decisions = decisionLog }) : "[]" | ||||||
|  | //        }; | ||||||
|  | 
 | ||||||
|  | //        await Client.Instance.From<GameAttempt>().Insert(gameAttempt); | ||||||
|  | //        Debug.Log("✅ Supabase Game Result Submitted"); | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    [Serializable] | ||||||
|  | //    public class Decision | ||||||
|  | //    { | ||||||
|  | //        public string decisionId; | ||||||
|  | //        public string timestamp; | ||||||
|  | //        public bool optimal; | ||||||
|  | //    } | ||||||
|  | 
 | ||||||
|  | //    [Serializable] | ||||||
|  | //    public class DecisionLogWrapper | ||||||
|  | //    { | ||||||
|  | //        public List<Decision> decisions; | ||||||
|  | //    } | ||||||
|  | //} | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ////using UnityEngine; | ||||||
|  | ////using System; | ||||||
|  | ////using System.Collections.Generic; | ||||||
|  | ////using System.Threading.Tasks; | ||||||
|  | ////using Supabase; // Make sure SupabaseManager initializes this correctly | ||||||
|  | ////using Postgrest.Models; | ||||||
|  | ////using Postgrest.Attributes; | ||||||
|  | 
 | ||||||
|  | ////public class SupabaseEventLogger : MonoBehaviour | ||||||
|  | ////{ | ||||||
|  | ////    public static SupabaseEventLogger Instance; | ||||||
|  | 
 | ||||||
|  | ////    private DateTime sessionStartTime; | ||||||
|  | 
 | ||||||
|  | ////    private void Awake() | ||||||
|  | ////    { | ||||||
|  | ////        if (Instance == null) | ||||||
|  | ////            Instance = this; | ||||||
|  | ////        else | ||||||
|  | ////            Destroy(gameObject); | ||||||
|  | ////    } | ||||||
|  | 
 | ||||||
|  | ////    /// <summary> | ||||||
|  | ////    /// Call this at the start of the game session. | ||||||
|  | ////    /// </summary> | ||||||
|  | ////    public async void StartSession() | ||||||
|  | ////    { | ||||||
|  | ////        sessionStartTime = DateTime.UtcNow; | ||||||
|  | 
 | ||||||
|  | ////        var gameEvent = new GameEvent | ||||||
|  | ////        { | ||||||
|  | ////            Id = Guid.NewGuid(), // <== Ensure this is explicitly set | ||||||
|  | ////            EventKey = "game_session_started", | ||||||
|  | ////            Timestamp = sessionStartTime, | ||||||
|  | ////            UserId = "user123" | ||||||
|  | ////        }; | ||||||
|  | ////        await Client.Instance.From<GameEvent>().Insert(gameEvent); | ||||||
|  | ////        Debug.Log("✅ Supabase Event: game_session_started"); | ||||||
|  | ////    } | ||||||
|  | 
 | ||||||
|  | ////    /// <summary> | ||||||
|  | ////    /// Logs optimal/suboptimal decisions at runtime. | ||||||
|  | ////    /// </summary> | ||||||
|  | ////    public async void LogDecisionEvent(bool isOptimal) | ||||||
|  | ////    { | ||||||
|  | ////        string eventKey = isOptimal ? "game_optimal_decision_made" : "game_suboptimal_decision_made"; | ||||||
|  | 
 | ||||||
|  | ////        var gameEvent = new GameEvent | ||||||
|  | ////        { | ||||||
|  | ////            Id = Guid.NewGuid(), | ||||||
|  | ////            EventKey = eventKey, | ||||||
|  | ////            Timestamp = DateTime.UtcNow, | ||||||
|  | ////            UserId = "user123" | ||||||
|  | ////        }; | ||||||
|  | 
 | ||||||
|  | ////        await Client.Instance.From<GameEvent>().Insert(gameEvent); | ||||||
|  | ////        Debug.Log($"✅ Supabase Event: {eventKey}"); | ||||||
|  | ////    } | ||||||
|  | 
 | ||||||
|  | ////    /// <summary> | ||||||
|  | ////    /// Completes the session and submits full results to phishing_game_attempts table. | ||||||
|  | ////    /// </summary> | ||||||
|  | ////    public async void CompleteSessionAndSubmitResult(string userId, bool passed, int optimal, int suboptimal, string scenarioId, List<Decision> decisionLog = null) | ||||||
|  | ////    { | ||||||
|  | ////        var endTime = DateTime.UtcNow; | ||||||
|  | ////        int duration = (int)(endTime - sessionStartTime).TotalSeconds; | ||||||
|  | 
 | ||||||
|  | ////        // Log completion events | ||||||
|  | ////        await Client.Instance.From<GameEvent>().Insert(new GameEvent | ||||||
|  | ////        { | ||||||
|  | ////            Id = Guid.NewGuid(), | ||||||
|  | ////            EventKey = "game_session_completed", | ||||||
|  | ////            Timestamp = endTime, | ||||||
|  | ////            UserId = userId | ||||||
|  | ////        }); | ||||||
|  | 
 | ||||||
|  | ////        await Client.Instance.From<GameEvent>().Insert(new GameEvent | ||||||
|  | ////        {Id = Guid.NewGuid(), | ||||||
|  | ////            EventKey = "game_score_recorded", | ||||||
|  | ////            Timestamp = endTime, | ||||||
|  | ////            UserId = userId | ||||||
|  | ////        }); | ||||||
|  | 
 | ||||||
|  | ////        // Insert session result | ||||||
|  | ////        var gameAttempt = new GameAttempt | ||||||
|  | ////        { | ||||||
|  | ////            GameId = "phishing-awareness-1", | ||||||
|  | ////            ScenarioId = scenarioId, | ||||||
|  | ////            UserId = userId, | ||||||
|  | ////            AttemptNumber = 1, | ||||||
|  | ////            StartTime = sessionStartTime, | ||||||
|  | ////            EndTime = endTime, | ||||||
|  | ////            DurationSeconds = duration, | ||||||
|  | ////            Score = passed ? 100 : 50, | ||||||
|  | ////            Passed = passed, | ||||||
|  | ////            Optimal = optimal, | ||||||
|  | ////            Suboptimal = suboptimal, | ||||||
|  | ////            KeyDecisionsLogJson = decisionLog != null ? JsonUtility.ToJson(new DecisionLogWrapper { decisions = decisionLog }) : "[]" | ||||||
|  | ////        }; | ||||||
|  | 
 | ||||||
|  | ////        await Client.Instance.From<GameAttempt>().Insert(gameAttempt); | ||||||
|  | ////        Debug.Log("✅ Supabase Game Result Submitted"); | ||||||
|  | ////    } | ||||||
|  | 
 | ||||||
|  | ////    [Serializable] | ||||||
|  | ////    public class Decision | ||||||
|  | ////    { | ||||||
|  | ////        public string decisionId; | ||||||
|  | ////        public string timestamp; | ||||||
|  | ////        public bool optimal; | ||||||
|  | ////    } | ||||||
|  | 
 | ||||||
|  | ////    [Serializable] | ||||||
|  | ////    public class DecisionLogWrapper | ||||||
|  | ////    { | ||||||
|  | ////        public List<Decision> decisions; | ||||||
|  | ////    } | ||||||
|  | ////} | ||||||
|  | 
 | ||||||
|  | ////// Existing SupabaseEventLogger class here... | ||||||
|  | ////// ↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓ | ||||||
|  | 
 | ||||||
|  | ////[Table("game_events")] | ||||||
|  | ////public class GameEvent : BaseModel | ||||||
|  | ////{ | ||||||
|  | ////    [PrimaryKey("id", false)] | ||||||
|  | ////    public Guid Id { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("event_key")] | ||||||
|  | ////    public string EventKey { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("timestamp")] | ||||||
|  | ////    public DateTime Timestamp { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("user_id")] | ||||||
|  | ////    public string UserId { get; set; } | ||||||
|  | ////} | ||||||
|  | 
 | ||||||
|  | ////[Table("phishing_game_attempts")] | ||||||
|  | ////public class GameAttempt : BaseModel | ||||||
|  | ////{ | ||||||
|  | ////    [PrimaryKey("id", false)] | ||||||
|  | ////    public Guid Id { get; set; } = Guid.NewGuid(); | ||||||
|  | 
 | ||||||
|  | ////    [Column("game_id")] | ||||||
|  | ////    public string GameId { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("scenario_id")] | ||||||
|  | ////    public string ScenarioId { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("user_id")] | ||||||
|  | ////    public string UserId { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("attempt_number")] | ||||||
|  | ////    public int AttemptNumber { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("start_timestamp")] | ||||||
|  | ////    public DateTime StartTime { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("end_timestamp")] | ||||||
|  | ////    public DateTime EndTime { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("duration_seconds")] | ||||||
|  | ////    public int DurationSeconds { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("final_score_percentage")] | ||||||
|  | ////    public float Score { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("pass_fail_status")] | ||||||
|  | ////    public bool Passed { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("optimal_decisions_count")] | ||||||
|  | ////    public int Optimal { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("suboptimal_decisions_count")] | ||||||
|  | ////    public int Suboptimal { get; set; } | ||||||
|  | 
 | ||||||
|  | ////    [Column("key_decisions_log")] | ||||||
|  | ////    public string KeyDecisionsLogJson { get; set; } | ||||||
|  | ////} | ||||||
|  | |||||||
| @ -816,14 +816,14 @@ PlayerSettings: | |||||||
|   webGLExceptionSupport: 1 |   webGLExceptionSupport: 1 | ||||||
|   webGLNameFilesAsHashes: 0 |   webGLNameFilesAsHashes: 0 | ||||||
|   webGLShowDiagnostics: 0 |   webGLShowDiagnostics: 0 | ||||||
|   webGLDataCaching: 1 |   webGLDataCaching: 0 | ||||||
|   webGLDebugSymbols: 0 |   webGLDebugSymbols: 0 | ||||||
|   webGLEmscriptenArgs:  |   webGLEmscriptenArgs:  | ||||||
|   webGLModulesDirectory:  |   webGLModulesDirectory:  | ||||||
|   webGLTemplate: APPLICATION:Default |   webGLTemplate: APPLICATION:Default | ||||||
|   webGLAnalyzeBuildSize: 0 |   webGLAnalyzeBuildSize: 0 | ||||||
|   webGLUseEmbeddedResources: 0 |   webGLUseEmbeddedResources: 0 | ||||||
|   webGLCompressionFormat: 1 |   webGLCompressionFormat: 2 | ||||||
|   webGLWasmArithmeticExceptions: 0 |   webGLWasmArithmeticExceptions: 0 | ||||||
|   webGLLinkerTarget: 1 |   webGLLinkerTarget: 1 | ||||||
|   webGLThreadsSupport: 0 |   webGLThreadsSupport: 0 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Ali Sharoz
						Ali Sharoz