From 0da8ebe1c1d3d7b3ce6218d33b4df4df9ba16995 Mon Sep 17 00:00:00 2001 From: gamer147 Date: Tue, 26 May 2026 14:16:32 -0400 Subject: [PATCH] refactor(bootstrap): migrate basic puzzles to seed files Replaces GlobalsImporter's ImportPuzzleGroups/Puzzles/Missions methods (plus the DeriveTargetPuzzleGroupId regex helper) with a dedicated PuzzleImporter that reads three flat seed JSONs (puzzle-groups, puzzles, puzzle-missions) produced by the Python extractor. Groups run before puzzles to satisfy the FK; missions upsert by sequential id. Wired into Program.cs and SVSimTestFactory after PaymentItemImporter so existing GlobalsImporterPuzzleTests continue to pass unchanged via SeedGlobalsAsync. The original prod-capture JSONs are deleted now that the seeds are authoritative. Co-Authored-By: Claude Opus 4.7 --- .../basic-puzzle-info-2026-05-23.json | 1 - .../basic-puzzle-mission-2026-05-23.json | 1 - SVSim.Bootstrap/Data/seeds/puzzle-groups.json | 305 ++++++ .../Data/seeds/puzzle-missions.json | 230 +++++ SVSim.Bootstrap/Data/seeds/puzzles.json | 906 ++++++++++++++++++ SVSim.Bootstrap/Importers/GlobalsImporter.cs | 148 --- SVSim.Bootstrap/Importers/PuzzleImporter.cs | 142 +++ .../Models/Seed/PuzzleGroupSeed.cs | 14 + .../Models/Seed/PuzzleMissionSeed.cs | 17 + SVSim.Bootstrap/Models/Seed/PuzzleSeed.cs | 13 + SVSim.Bootstrap/Program.cs | 4 + .../Importers/PuzzleImporterTests.cs | 154 +++ .../Infrastructure/SVSimTestFactory.cs | 4 + 13 files changed, 1789 insertions(+), 150 deletions(-) delete mode 100644 SVSim.Bootstrap/Data/prod-captures/basic-puzzle-info-2026-05-23.json delete mode 100644 SVSim.Bootstrap/Data/prod-captures/basic-puzzle-mission-2026-05-23.json create mode 100644 SVSim.Bootstrap/Data/seeds/puzzle-groups.json create mode 100644 SVSim.Bootstrap/Data/seeds/puzzle-missions.json create mode 100644 SVSim.Bootstrap/Data/seeds/puzzles.json create mode 100644 SVSim.Bootstrap/Importers/PuzzleImporter.cs create mode 100644 SVSim.Bootstrap/Models/Seed/PuzzleGroupSeed.cs create mode 100644 SVSim.Bootstrap/Models/Seed/PuzzleMissionSeed.cs create mode 100644 SVSim.Bootstrap/Models/Seed/PuzzleSeed.cs create mode 100644 SVSim.UnitTests/Importers/PuzzleImporterTests.cs diff --git a/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-info-2026-05-23.json b/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-info-2026-05-23.json deleted file mode 100644 index 30fb69f..0000000 --- a/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-info-2026-05-23.json +++ /dev/null @@ -1 +0,0 @@ -{"data_headers":{"sid":"41600731e6a1097b3c319a524bd73faf1779643058","short_udid":411054851,"viewer_id":906243102,"servertime":1779643058,"result_code":1},"data":[{"puzzle_master_id":"316","puzzle_data":[{"puzzle_id":"106","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"107","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"108","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"109","puzzle_difficulty":"2","is_cleared":false,"is_additional":true,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"110","puzzle_difficulty":"2","is_cleared":false,"is_additional":true,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"111","puzzle_difficulty":"2","is_cleared":false,"is_additional":true,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"112","puzzle_difficulty":"2","is_cleared":false,"is_additional":true,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"113","puzzle_difficulty":"2","is_cleared":false,"is_additional":true,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0316","is_mission_target":true},{"puzzle_master_id":"315","puzzle_data":[{"puzzle_id":"103","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"104","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"105","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0315","is_mission_target":true},{"puzzle_master_id":"314","puzzle_data":[{"puzzle_id":"100","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"101","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"102","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0314","is_mission_target":true},{"puzzle_master_id":"313","puzzle_data":[{"puzzle_id":"97","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"98","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"99","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0313","is_mission_target":true},{"puzzle_master_id":"312","puzzle_data":[{"puzzle_id":"94","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"95","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"96","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0312","is_mission_target":true},{"puzzle_master_id":"311","puzzle_data":[{"puzzle_id":"91","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"92","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"93","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0311","is_mission_target":true},{"puzzle_master_id":"310","puzzle_data":[{"puzzle_id":"84","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"85","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"86","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0310","is_mission_target":true},{"puzzle_master_id":"309","puzzle_data":[{"puzzle_id":"77","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"78","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"79","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0309","is_mission_target":true},{"puzzle_master_id":"308","puzzle_data":[{"puzzle_id":"74","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"75","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"76","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0308","is_mission_target":true},{"puzzle_master_id":"307","puzzle_data":[{"puzzle_id":"67","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"68","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"69","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0307","is_mission_target":true},{"puzzle_master_id":"306","puzzle_data":[{"puzzle_id":"64","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"65","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"66","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0306","is_mission_target":true},{"puzzle_master_id":"305","puzzle_data":[{"puzzle_id":"61","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"62","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"63","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0305","is_mission_target":true},{"puzzle_master_id":"304","puzzle_data":[{"puzzle_id":"46","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"47","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"48","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0304","is_mission_target":true},{"puzzle_master_id":"303","puzzle_data":[{"puzzle_id":"43","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"44","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"45","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0303","is_mission_target":true},{"puzzle_master_id":"302","puzzle_data":[{"puzzle_id":"40","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"41","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"42","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0302","is_mission_target":true},{"puzzle_master_id":"301","puzzle_data":[{"puzzle_id":"37","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"38","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"39","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3704","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3704","sort_type":"1","basic_title_text_id":"Puzzle_QuestSelect_0301","is_mission_target":true},{"puzzle_master_id":"9","puzzle_data":[{"puzzle_id":"87","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"88","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"89","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"90","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"600090","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2","":"3"},"is_all_cleared":false,"chara_id":"600090","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0109","is_mission_target":false},{"puzzle_master_id":"8","puzzle_data":[{"puzzle_id":"80","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"81","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"82","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"83","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"600080","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2","":"3"},"is_all_cleared":false,"chara_id":"600080","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0108","is_mission_target":false},{"puzzle_master_id":"7","puzzle_data":[{"puzzle_id":"70","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"71","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"72","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"73","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"600070","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"600070","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0107","is_mission_target":false},{"puzzle_master_id":"6","puzzle_data":[{"puzzle_id":"52","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"53","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"54","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"55","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"56","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"57","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"58","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"59","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"60","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"600060","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2","":"3"},"is_all_cleared":false,"chara_id":"600060","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0106","is_mission_target":false},{"puzzle_master_id":"5","puzzle_data":[{"puzzle_id":"49","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"50","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"51","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3801","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3801","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0105","is_mission_target":false},{"puzzle_master_id":"4","puzzle_data":[{"puzzle_id":"28","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"29","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"30","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"31","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"32","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"33","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"34","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"35","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"36","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3603","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2","":"3"},"is_all_cleared":false,"chara_id":"3603","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0104","is_mission_target":false},{"puzzle_master_id":"3","puzzle_data":[{"puzzle_id":"20","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"21","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"22","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"23","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"24","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"25","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"26","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"27","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"3403","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"3403","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0103","is_mission_target":false},{"puzzle_master_id":"2","puzzle_data":[{"puzzle_id":"10","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"11","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"12","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"13","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"14","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"15","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"16","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"17","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":"Puzzle_Unlock_Condition_0001"},{"puzzle_id":"18","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":"Puzzle_Unlock_Condition_0001"},{"puzzle_id":"19","puzzle_difficulty":"3","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":"Puzzle_Unlock_Condition_0001"}],"puzzle_chara_id":"3208","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2","":"3"},"is_all_cleared":false,"chara_id":"2703","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0102","is_mission_target":false},{"puzzle_master_id":"1","puzzle_data":[{"puzzle_id":"1","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"2","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"3","puzzle_difficulty":"0","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"4","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"5","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"6","puzzle_difficulty":"1","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"7","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"8","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""},{"puzzle_id":"9","puzzle_difficulty":"2","is_cleared":false,"is_additional":false,"is_playable":true,"release_condition_text_id":""}],"puzzle_chara_id":"600050","puzzle_difficulty_name_list":{"Beginner":"0","Experienced":"1","Expert":"2"},"is_all_cleared":false,"chara_id":"600050","sort_type":"2","basic_title_text_id":"Puzzle_QuestSelect_0101","is_mission_target":false}]} diff --git a/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-mission-2026-05-23.json b/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-mission-2026-05-23.json deleted file mode 100644 index 0686c10..0000000 --- a/SVSim.Bootstrap/Data/prod-captures/basic-puzzle-mission-2026-05-23.json +++ /dev/null @@ -1 +0,0 @@ -{"data_headers":{"sid":"079f239bb83de281ebc6b2f68dbb2cd11779683743","short_udid":411054851,"viewer_id":906243102,"servertime":1779683743,"result_code":1},"data":[{"mission_name":"Clear all Dragoncraft and Portalcraft puzzles puzzles in the Special Round","require_number":"2","campaign_commence_time":1725670800,"reward_list":[{"reward_type":"4","reward_detail_id":"90001","reward_number":"1"}],"order_id":"5","total_count":"0","is_achieved":false},{"mission_name":"Clear all Forestcraft, Shadowcraft and Bloodcraft puzzles in the Special Round","require_number":"3","campaign_commence_time":1722646800,"reward_list":[{"reward_type":"4","reward_detail_id":"90001","reward_number":"1"}],"order_id":"4","total_count":"0","is_achieved":false},{"mission_name":"Clear all Swordcraft, Runecraft and Havencraft puzzles in the Special Round","require_number":"3","campaign_commence_time":1720227600,"reward_list":[{"reward_type":"4","reward_detail_id":"90001","reward_number":"1"}],"order_id":"3","total_count":"0","is_achieved":false},{"mission_name":"Clear all Special Round puzzles","require_number":"8","campaign_commence_time":1720227600,"reward_list":[{"reward_type":"7","reward_detail_id":"400004315","reward_number":"1"}],"order_id":"2","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 15 puzzles","require_number":"3","campaign_commence_time":1716598800,"reward_list":[{"reward_type":"7","reward_detail_id":"400004314","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 14 puzzles","require_number":"3","campaign_commence_time":1711760400,"reward_list":[{"reward_type":"6","reward_detail_id":"3065004","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 13 puzzles","require_number":"3","campaign_commence_time":1708736400,"reward_list":[{"reward_type":"7","reward_detail_id":"400004313","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 12 puzzles","require_number":"3","campaign_commence_time":1703898000,"reward_list":[{"reward_type":"6","reward_detail_id":"3074009","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 11 puzzles","require_number":"3","campaign_commence_time":1700269200,"reward_list":[{"reward_type":"6","reward_detail_id":"3074008","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 10 puzzles","require_number":"3","campaign_commence_time":1692406800,"reward_list":[{"reward_type":"6","reward_detail_id":"3074007","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 9 puzzles","require_number":"3","campaign_commence_time":1688173200,"reward_list":[{"reward_type":"6","reward_detail_id":"3074006","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 8 puzzles","require_number":"3","campaign_commence_time":1684544400,"reward_list":[{"reward_type":"6","reward_detail_id":"3074005","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 7 puzzles","require_number":"3","campaign_commence_time":1677286800,"reward_list":[{"reward_type":"6","reward_detail_id":"3074004","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 6 puzzles","require_number":"3","campaign_commence_time":1672448400,"reward_list":[{"reward_type":"6","reward_detail_id":"3074003","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 5 puzzles","require_number":"3","campaign_commence_time":1669424400,"reward_list":[{"reward_type":"6","reward_detail_id":"3074002","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 4 puzzles","require_number":"3","campaign_commence_time":1660959000,"reward_list":[{"reward_type":"6","reward_detail_id":"3074001","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 3 puzzles","require_number":"3","campaign_commence_time":1656725400,"reward_list":[{"reward_type":"7","reward_detail_id":"400004105","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 2 puzzles","require_number":"3","campaign_commence_time":1653096600,"reward_list":[{"reward_type":"7","reward_detail_id":"400004104","reward_number":"1"}],"order_id":"1","total_count":"0","is_achieved":false},{"mission_name":"Clear all Round 1 puzzles","require_number":"3","campaign_commence_time":1651282200,"reward_list":[{"reward_type":"10","reward_detail_id":"3704","reward_number":"1"}],"order_id":"1","total_count":"3","is_achieved":true}]} diff --git a/SVSim.Bootstrap/Data/seeds/puzzle-groups.json b/SVSim.Bootstrap/Data/seeds/puzzle-groups.json new file mode 100644 index 0000000..b5733ac --- /dev/null +++ b/SVSim.Bootstrap/Data/seeds/puzzle-groups.json @@ -0,0 +1,305 @@ +[ + { + "id": 316, + "basic_title_text_id": "Puzzle_QuestSelect_0316", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Expert": "2" + } + }, + { + "id": 315, + "basic_title_text_id": "Puzzle_QuestSelect_0315", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 314, + "basic_title_text_id": "Puzzle_QuestSelect_0314", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 313, + "basic_title_text_id": "Puzzle_QuestSelect_0313", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 312, + "basic_title_text_id": "Puzzle_QuestSelect_0312", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 311, + "basic_title_text_id": "Puzzle_QuestSelect_0311", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 310, + "basic_title_text_id": "Puzzle_QuestSelect_0310", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 309, + "basic_title_text_id": "Puzzle_QuestSelect_0309", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 308, + "basic_title_text_id": "Puzzle_QuestSelect_0308", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 307, + "basic_title_text_id": "Puzzle_QuestSelect_0307", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 306, + "basic_title_text_id": "Puzzle_QuestSelect_0306", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 305, + "basic_title_text_id": "Puzzle_QuestSelect_0305", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 304, + "basic_title_text_id": "Puzzle_QuestSelect_0304", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 303, + "basic_title_text_id": "Puzzle_QuestSelect_0303", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 302, + "basic_title_text_id": "Puzzle_QuestSelect_0302", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 301, + "basic_title_text_id": "Puzzle_QuestSelect_0301", + "puzzle_chara_id": 3704, + "chara_id": 3704, + "sort_type": 1, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 9, + "basic_title_text_id": "Puzzle_QuestSelect_0109", + "puzzle_chara_id": 600090, + "chara_id": 600090, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2", + "": "3" + } + }, + { + "id": 8, + "basic_title_text_id": "Puzzle_QuestSelect_0108", + "puzzle_chara_id": 600080, + "chara_id": 600080, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2", + "": "3" + } + }, + { + "id": 7, + "basic_title_text_id": "Puzzle_QuestSelect_0107", + "puzzle_chara_id": 600070, + "chara_id": 600070, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 6, + "basic_title_text_id": "Puzzle_QuestSelect_0106", + "puzzle_chara_id": 600060, + "chara_id": 600060, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2", + "": "3" + } + }, + { + "id": 5, + "basic_title_text_id": "Puzzle_QuestSelect_0105", + "puzzle_chara_id": 3801, + "chara_id": 3801, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 4, + "basic_title_text_id": "Puzzle_QuestSelect_0104", + "puzzle_chara_id": 3603, + "chara_id": 3603, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2", + "": "3" + } + }, + { + "id": 3, + "basic_title_text_id": "Puzzle_QuestSelect_0103", + "puzzle_chara_id": 3403, + "chara_id": 3403, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + }, + { + "id": 2, + "basic_title_text_id": "Puzzle_QuestSelect_0102", + "puzzle_chara_id": 3208, + "chara_id": 2703, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2", + "": "3" + } + }, + { + "id": 1, + "basic_title_text_id": "Puzzle_QuestSelect_0101", + "puzzle_chara_id": 600050, + "chara_id": 600050, + "sort_type": 2, + "difficulty_name_list": { + "Beginner": "0", + "Experienced": "1", + "Expert": "2" + } + } +] diff --git a/SVSim.Bootstrap/Data/seeds/puzzle-missions.json b/SVSim.Bootstrap/Data/seeds/puzzle-missions.json new file mode 100644 index 0000000..ac236bd --- /dev/null +++ b/SVSim.Bootstrap/Data/seeds/puzzle-missions.json @@ -0,0 +1,230 @@ +[ + { + "id": 1, + "mission_name": "Clear all Dragoncraft and Portalcraft puzzles puzzles in the Special Round", + "achieved_message": "Mission achieved", + "require_number": 2, + "campaign_commence_time": 1725670800, + "order_id": 5, + "reward_type": 4, + "reward_detail_id": 90001, + "reward_number": 1, + "target_puzzle_group_id": null + }, + { + "id": 2, + "mission_name": "Clear all Forestcraft, Shadowcraft and Bloodcraft puzzles in the Special Round", + "achieved_message": "Mission achieved", + "require_number": 3, + "campaign_commence_time": 1722646800, + "order_id": 4, + "reward_type": 4, + "reward_detail_id": 90001, + "reward_number": 1, + "target_puzzle_group_id": null + }, + { + "id": 3, + "mission_name": "Clear all Swordcraft, Runecraft and Havencraft puzzles in the Special Round", + "achieved_message": "Mission achieved", + "require_number": 3, + "campaign_commence_time": 1720227600, + "order_id": 3, + "reward_type": 4, + "reward_detail_id": 90001, + "reward_number": 1, + "target_puzzle_group_id": null + }, + { + "id": 4, + "mission_name": "Clear all Special Round puzzles", + "achieved_message": "Mission achieved", + "require_number": 8, + "campaign_commence_time": 1720227600, + "order_id": 2, + "reward_type": 7, + "reward_detail_id": 400004315, + "reward_number": 1, + "target_puzzle_group_id": null + }, + { + "id": 5, + "mission_name": "Clear all Round 15 puzzles", + "achieved_message": "Cleared all Round 15 puzzles", + "require_number": 3, + "campaign_commence_time": 1716598800, + "order_id": 1, + "reward_type": 7, + "reward_detail_id": 400004314, + "reward_number": 1, + "target_puzzle_group_id": 315 + }, + { + "id": 6, + "mission_name": "Clear all Round 14 puzzles", + "achieved_message": "Cleared all Round 14 puzzles", + "require_number": 3, + "campaign_commence_time": 1711760400, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3065004, + "reward_number": 1, + "target_puzzle_group_id": 314 + }, + { + "id": 7, + "mission_name": "Clear all Round 13 puzzles", + "achieved_message": "Cleared all Round 13 puzzles", + "require_number": 3, + "campaign_commence_time": 1708736400, + "order_id": 1, + "reward_type": 7, + "reward_detail_id": 400004313, + "reward_number": 1, + "target_puzzle_group_id": 313 + }, + { + "id": 8, + "mission_name": "Clear all Round 12 puzzles", + "achieved_message": "Cleared all Round 12 puzzles", + "require_number": 3, + "campaign_commence_time": 1703898000, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074009, + "reward_number": 1, + "target_puzzle_group_id": 312 + }, + { + "id": 9, + "mission_name": "Clear all Round 11 puzzles", + "achieved_message": "Cleared all Round 11 puzzles", + "require_number": 3, + "campaign_commence_time": 1700269200, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074008, + "reward_number": 1, + "target_puzzle_group_id": 311 + }, + { + "id": 10, + "mission_name": "Clear all Round 10 puzzles", + "achieved_message": "Cleared all Round 10 puzzles", + "require_number": 3, + "campaign_commence_time": 1692406800, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074007, + "reward_number": 1, + "target_puzzle_group_id": 310 + }, + { + "id": 11, + "mission_name": "Clear all Round 9 puzzles", + "achieved_message": "Cleared all Round 9 puzzles", + "require_number": 3, + "campaign_commence_time": 1688173200, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074006, + "reward_number": 1, + "target_puzzle_group_id": 309 + }, + { + "id": 12, + "mission_name": "Clear all Round 8 puzzles", + "achieved_message": "Cleared all Round 8 puzzles", + "require_number": 3, + "campaign_commence_time": 1684544400, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074005, + "reward_number": 1, + "target_puzzle_group_id": 308 + }, + { + "id": 13, + "mission_name": "Clear all Round 7 puzzles", + "achieved_message": "Cleared all Round 7 puzzles", + "require_number": 3, + "campaign_commence_time": 1677286800, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074004, + "reward_number": 1, + "target_puzzle_group_id": 307 + }, + { + "id": 14, + "mission_name": "Clear all Round 6 puzzles", + "achieved_message": "Cleared all Round 6 puzzles", + "require_number": 3, + "campaign_commence_time": 1672448400, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074003, + "reward_number": 1, + "target_puzzle_group_id": 306 + }, + { + "id": 15, + "mission_name": "Clear all Round 5 puzzles", + "achieved_message": "Cleared all Round 5 puzzles", + "require_number": 3, + "campaign_commence_time": 1669424400, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074002, + "reward_number": 1, + "target_puzzle_group_id": 305 + }, + { + "id": 16, + "mission_name": "Clear all Round 4 puzzles", + "achieved_message": "Cleared all Round 4 puzzles", + "require_number": 3, + "campaign_commence_time": 1660959000, + "order_id": 1, + "reward_type": 6, + "reward_detail_id": 3074001, + "reward_number": 1, + "target_puzzle_group_id": 304 + }, + { + "id": 17, + "mission_name": "Clear all Round 3 puzzles", + "achieved_message": "Cleared all Round 3 puzzles", + "require_number": 3, + "campaign_commence_time": 1656725400, + "order_id": 1, + "reward_type": 7, + "reward_detail_id": 400004105, + "reward_number": 1, + "target_puzzle_group_id": 303 + }, + { + "id": 18, + "mission_name": "Clear all Round 2 puzzles", + "achieved_message": "Cleared all Round 2 puzzles", + "require_number": 3, + "campaign_commence_time": 1653096600, + "order_id": 1, + "reward_type": 7, + "reward_detail_id": 400004104, + "reward_number": 1, + "target_puzzle_group_id": 302 + }, + { + "id": 19, + "mission_name": "Clear all Round 1 puzzles", + "achieved_message": "Cleared all Round 1 puzzles", + "require_number": 3, + "campaign_commence_time": 1651282200, + "order_id": 1, + "reward_type": 10, + "reward_detail_id": 3704, + "reward_number": 1, + "target_puzzle_group_id": 301 + } +] diff --git a/SVSim.Bootstrap/Data/seeds/puzzles.json b/SVSim.Bootstrap/Data/seeds/puzzles.json new file mode 100644 index 0000000..9b1d52e --- /dev/null +++ b/SVSim.Bootstrap/Data/seeds/puzzles.json @@ -0,0 +1,906 @@ +[ + { + "id": 106, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 107, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 108, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 109, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": true, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 110, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": true, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 111, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": true, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 112, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": true, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 113, + "group_id": 316, + "puzzle_difficulty": 2, + "is_additional": true, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 103, + "group_id": 315, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 104, + "group_id": 315, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 105, + "group_id": 315, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 100, + "group_id": 314, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 101, + "group_id": 314, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 102, + "group_id": 314, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 97, + "group_id": 313, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 98, + "group_id": 313, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 99, + "group_id": 313, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 94, + "group_id": 312, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 95, + "group_id": 312, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 96, + "group_id": 312, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 91, + "group_id": 311, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 92, + "group_id": 311, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 93, + "group_id": 311, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 84, + "group_id": 310, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 85, + "group_id": 310, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 86, + "group_id": 310, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 77, + "group_id": 309, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 78, + "group_id": 309, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 79, + "group_id": 309, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 74, + "group_id": 308, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 75, + "group_id": 308, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 76, + "group_id": 308, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 67, + "group_id": 307, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 68, + "group_id": 307, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 69, + "group_id": 307, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 64, + "group_id": 306, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 65, + "group_id": 306, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 66, + "group_id": 306, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 61, + "group_id": 305, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 62, + "group_id": 305, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 63, + "group_id": 305, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 46, + "group_id": 304, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 47, + "group_id": 304, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 48, + "group_id": 304, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 43, + "group_id": 303, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 44, + "group_id": 303, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 45, + "group_id": 303, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 40, + "group_id": 302, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 41, + "group_id": 302, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 42, + "group_id": 302, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 37, + "group_id": 301, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 38, + "group_id": 301, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 39, + "group_id": 301, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 87, + "group_id": 9, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 88, + "group_id": 9, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 89, + "group_id": 9, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 90, + "group_id": 9, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 80, + "group_id": 8, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 81, + "group_id": 8, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 82, + "group_id": 8, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 83, + "group_id": 8, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 70, + "group_id": 7, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 71, + "group_id": 7, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 72, + "group_id": 7, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 73, + "group_id": 7, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 52, + "group_id": 6, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 53, + "group_id": 6, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 54, + "group_id": 6, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 55, + "group_id": 6, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 56, + "group_id": 6, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 57, + "group_id": 6, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 58, + "group_id": 6, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 59, + "group_id": 6, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 60, + "group_id": 6, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 49, + "group_id": 5, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 50, + "group_id": 5, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 51, + "group_id": 5, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 28, + "group_id": 4, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 29, + "group_id": 4, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 30, + "group_id": 4, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 31, + "group_id": 4, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 32, + "group_id": 4, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 33, + "group_id": 4, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 34, + "group_id": 4, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 35, + "group_id": 4, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 36, + "group_id": 4, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 20, + "group_id": 3, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 21, + "group_id": 3, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 22, + "group_id": 3, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 23, + "group_id": 3, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 24, + "group_id": 3, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 25, + "group_id": 3, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 26, + "group_id": 3, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 27, + "group_id": 3, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 10, + "group_id": 2, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 11, + "group_id": 2, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 12, + "group_id": 2, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 13, + "group_id": 2, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 14, + "group_id": 2, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 15, + "group_id": 2, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 16, + "group_id": 2, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 17, + "group_id": 2, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "Puzzle_Unlock_Condition_0001" + }, + { + "id": 18, + "group_id": 2, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "Puzzle_Unlock_Condition_0001" + }, + { + "id": 19, + "group_id": 2, + "puzzle_difficulty": 3, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "Puzzle_Unlock_Condition_0001" + }, + { + "id": 1, + "group_id": 1, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 2, + "group_id": 1, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 3, + "group_id": 1, + "puzzle_difficulty": 0, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 4, + "group_id": 1, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 5, + "group_id": 1, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 6, + "group_id": 1, + "puzzle_difficulty": 1, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 7, + "group_id": 1, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 8, + "group_id": 1, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + }, + { + "id": 9, + "group_id": 1, + "puzzle_difficulty": 2, + "is_additional": false, + "is_playable": true, + "release_condition_text_id": "" + } +] diff --git a/SVSim.Bootstrap/Importers/GlobalsImporter.cs b/SVSim.Bootstrap/Importers/GlobalsImporter.cs index 29ede7e..ebaed5e 100644 --- a/SVSim.Bootstrap/Importers/GlobalsImporter.cs +++ b/SVSim.Bootstrap/Importers/GlobalsImporter.cs @@ -1,5 +1,4 @@ using System.Text.Json; -using System.Text.RegularExpressions; using Microsoft.EntityFrameworkCore; using SVSim.Database; using SVSim.Database.Enums; @@ -31,8 +30,6 @@ public class GlobalsImporter JsonElement? mypageIndex = LoadCapture(capturesDir, "mypage-index"); JsonElement? deckInfo = LoadCapture(capturesDir, "deck-info"); JsonElement? packInfo = LoadCapture(capturesDir, "pack-info"); - JsonElement? basicPuzzleInfo = LoadCapture(capturesDir, "basic-puzzle-info"); - JsonElement? basicPuzzleMission = LoadCapture(capturesDir, "basic-puzzle-mission"); int total = 0; @@ -73,17 +70,6 @@ public class GlobalsImporter total += await ImportPacks(context, packInfo.Value); } - if (basicPuzzleInfo.HasValue) - { - total += await ImportPuzzleGroups(context, basicPuzzleInfo.Value); - total += await ImportPuzzles(context, basicPuzzleInfo.Value); - } - - if (basicPuzzleMission.HasValue) - { - total += await ImportPuzzleMissions(context, basicPuzzleMission.Value); - } - await context.SaveChangesAsync(); Console.WriteLine($"[GlobalsImporter] Done: {total} total rows changed."); return total; @@ -819,140 +805,6 @@ public class GlobalsImporter return created + updated; } - // ---------- Basic Puzzle Groups + Puzzles ---------- - - /// - /// /basic_puzzle/info capture is an array of group objects keyed on puzzle_master_id. - /// Numeric wire fields come through as strings — GetInt tolerates both. Idempotent upsert - /// by puzzle_master_id; rows missing from a partial capture are left intact. - /// - private async Task ImportPuzzleGroups(SVSimDbContext context, JsonElement infoData) - { - if (infoData.ValueKind != JsonValueKind.Array) return 0; - - var existing = await context.PuzzleGroups.ToDictionaryAsync(e => e.Id); - int created = 0, updated = 0; - - foreach (var row in infoData.EnumerateArray()) - { - int masterId = GetInt(row, "puzzle_master_id"); - if (masterId == 0) continue; - - var entry = existing.TryGetValue(masterId, out var ex) ? ex : new PuzzleGroupEntry { Id = masterId }; - entry.BasicTitleTextId = GetString(row, "basic_title_text_id"); - entry.PuzzleCharaId = GetInt(row, "puzzle_chara_id"); - entry.CharaId = GetInt(row, "chara_id"); - entry.SortType = GetInt(row, "sort_type"); - entry.DifficultyNameListJson = row.TryGetProperty("puzzle_difficulty_name_list", out var d) - ? Serialize(d) - : "{}"; - - if (ex is null) { context.PuzzleGroups.Add(entry); created++; } - else updated++; - } - - Console.WriteLine($"[GlobalsImporter] PuzzleGroups: +{created}/~{updated}"); - return created + updated; - } - - /// - /// Walks each group's puzzle_data array and upserts PuzzleEntry rows keyed on puzzle_id. - /// Groups must have been imported first (FK PuzzleEntry.GroupId → PuzzleGroupEntry.Id). - /// - private async Task ImportPuzzles(SVSimDbContext context, JsonElement infoData) - { - if (infoData.ValueKind != JsonValueKind.Array) return 0; - - var existing = await context.Puzzles.ToDictionaryAsync(e => e.Id); - int created = 0, updated = 0; - - foreach (var group in infoData.EnumerateArray()) - { - int masterId = GetInt(group, "puzzle_master_id"); - if (masterId == 0 || !group.TryGetProperty("puzzle_data", out var puzzleArray)) continue; - if (puzzleArray.ValueKind != JsonValueKind.Array) continue; - - foreach (var p in puzzleArray.EnumerateArray()) - { - int puzzleId = GetInt(p, "puzzle_id"); - if (puzzleId == 0) continue; - - var entry = existing.TryGetValue(puzzleId, out var ex) ? ex : new PuzzleEntry { Id = puzzleId }; - entry.GroupId = masterId; - entry.PuzzleDifficulty = GetInt(p, "puzzle_difficulty"); - entry.IsAdditional = GetBool(p, "is_additional"); - entry.IsPlayable = GetBool(p, "is_playable"); - entry.ReleaseConditionTextId = GetString(p, "release_condition_text_id"); - - if (ex is null) { context.Puzzles.Add(entry); created++; } - else updated++; - } - } - - Console.WriteLine($"[GlobalsImporter] Puzzles: +{created}/~{updated}"); - return created + updated; - } - - // ---------- Basic Puzzle Missions ---------- - - private static readonly Regex RoundMissionPattern = - new(@"^Clear all Round (\d+) puzzles$", RegexOptions.Compiled); - - /// Maps the captured mission_name to its target puzzle_master_id. Returns null for - /// Special-Round entries — Phase 1 surfaces them with total_count=0 (see design § Out of Scope). - internal static int? DeriveTargetPuzzleGroupId(string missionName) - { - var m = RoundMissionPattern.Match(missionName); - return m.Success ? 300 + int.Parse(m.Groups[1].Value) : null; - } - - private async Task ImportPuzzleMissions(SVSimDbContext context, JsonElement missionData) - { - if (missionData.ValueKind != JsonValueKind.Array) return 0; - - // Key by 1-based sequence (the wire has no stable id); first run inserts, re-runs match by index. - var existing = await context.PuzzleMissions.ToDictionaryAsync(e => e.Id); - int created = 0, updated = 0, unmapped = 0; - - int seq = 1; - foreach (var row in missionData.EnumerateArray()) - { - string name = GetString(row, "mission_name"); - if (string.IsNullOrEmpty(name)) { seq++; continue; } - - var entry = existing.TryGetValue(seq, out var ex) ? ex : new PuzzleMissionEntry { Id = seq }; - entry.MissionName = name; - entry.AchievedMessage = RoundMissionPattern.IsMatch(name) - ? RoundMissionPattern.Replace(name, m => $"Cleared all Round {m.Groups[1].Value} puzzles") - : "Mission achieved"; // Special-Round fallback; only surfaces if a Special mission ever flips, which won't in Phase 1. - entry.RequireNumber = GetInt(row, "require_number"); - entry.CampaignCommenceTime = GetLong(row, "campaign_commence_time"); - entry.OrderId = GetInt(row, "order_id"); - - // reward_list[0] — single reward per mission. Skip if missing/empty. - if (row.TryGetProperty("reward_list", out var rl) && rl.ValueKind == JsonValueKind.Array && rl.GetArrayLength() > 0) - { - var r = rl[0]; - entry.RewardType = GetInt(r, "reward_type"); - entry.RewardDetailId = GetLong(r, "reward_detail_id"); - entry.RewardNumber = GetInt(r, "reward_number"); - } - - entry.TargetPuzzleGroupId = DeriveTargetPuzzleGroupId(name); - if (entry.TargetPuzzleGroupId is null) unmapped++; - - if (ex is null) { context.PuzzleMissions.Add(entry); created++; } - else updated++; - - seq++; - } - - if (unmapped > 0) - Console.WriteLine($"[GlobalsImporter] PuzzleMissions: {unmapped} Special-Round missions left unmapped (Phase 1 deferral)."); - Console.WriteLine($"[GlobalsImporter] PuzzleMissions: +{created}/~{updated}"); - return created + updated; - } - // ---------- Helpers ---------- private static void WarnOrphans(string label, int count) diff --git a/SVSim.Bootstrap/Importers/PuzzleImporter.cs b/SVSim.Bootstrap/Importers/PuzzleImporter.cs new file mode 100644 index 0000000..15dceb8 --- /dev/null +++ b/SVSim.Bootstrap/Importers/PuzzleImporter.cs @@ -0,0 +1,142 @@ +using System.Text.Json; +using Microsoft.EntityFrameworkCore; +using SVSim.Bootstrap.Models.Seed; +using SVSim.Database; +using SVSim.Database.Models; + +namespace SVSim.Bootstrap.Importers; + +/// +/// Idempotent upsert of the basic-puzzle catalog from seeds/puzzle-groups.json, +/// seeds/puzzles.json, and seeds/puzzle-missions.json. Groups must be imported +/// before puzzles (FK on -> ). +/// Rows missing from the seed are LEFT INTACT (consistent with other per-importer seeds). +/// +public class PuzzleImporter +{ + public async Task ImportGroupsAsync(SVSimDbContext context, string seedDir) + { + string path = Path.Combine(seedDir, "puzzle-groups.json"); + var seed = SeedLoader.LoadList(path); + if (seed.Count == 0) + { + Console.WriteLine("[PuzzleImporter] No group seed rows; skipping."); + return 0; + } + + var existing = await context.PuzzleGroups.ToDictionaryAsync(e => e.Id); + int created = 0, updated = 0; + + foreach (var s in seed) + { + if (s.Id == 0) continue; + + var entry = existing.TryGetValue(s.Id, out var ex) + ? ex : new PuzzleGroupEntry { Id = s.Id }; + + entry.BasicTitleTextId = s.BasicTitleTextId; + entry.PuzzleCharaId = s.PuzzleCharaId; + entry.CharaId = s.CharaId; + entry.SortType = s.SortType; + entry.DifficultyNameListJson = s.DifficultyNameList.ValueKind == JsonValueKind.Undefined + ? "{}" + : JsonSerializer.Serialize(s.DifficultyNameList); + + if (ex is null) + { + context.PuzzleGroups.Add(entry); + existing[s.Id] = entry; + created++; + } + else updated++; + } + + await context.SaveChangesAsync(); + Console.WriteLine($"[PuzzleImporter] Groups +{created}/~{updated}"); + return created + updated; + } + + public async Task ImportPuzzlesAsync(SVSimDbContext context, string seedDir) + { + string path = Path.Combine(seedDir, "puzzles.json"); + var seed = SeedLoader.LoadList(path); + if (seed.Count == 0) + { + Console.WriteLine("[PuzzleImporter] No puzzle seed rows; skipping."); + return 0; + } + + var existing = await context.Puzzles.ToDictionaryAsync(e => e.Id); + int created = 0, updated = 0; + + foreach (var s in seed) + { + if (s.Id == 0) continue; + + var entry = existing.TryGetValue(s.Id, out var ex) + ? ex : new PuzzleEntry { Id = s.Id }; + + entry.GroupId = s.GroupId; + entry.PuzzleDifficulty = s.PuzzleDifficulty; + entry.IsAdditional = s.IsAdditional; + entry.IsPlayable = s.IsPlayable; + entry.ReleaseConditionTextId = s.ReleaseConditionTextId; + + if (ex is null) + { + context.Puzzles.Add(entry); + existing[s.Id] = entry; + created++; + } + else updated++; + } + + await context.SaveChangesAsync(); + Console.WriteLine($"[PuzzleImporter] Puzzles +{created}/~{updated}"); + return created + updated; + } + + public async Task ImportMissionsAsync(SVSimDbContext context, string seedDir) + { + string path = Path.Combine(seedDir, "puzzle-missions.json"); + var seed = SeedLoader.LoadList(path); + if (seed.Count == 0) + { + Console.WriteLine("[PuzzleImporter] No mission seed rows; skipping."); + return 0; + } + + var existing = await context.PuzzleMissions.ToDictionaryAsync(e => e.Id); + int created = 0, updated = 0; + + foreach (var s in seed) + { + if (s.Id == 0) continue; + + var entry = existing.TryGetValue(s.Id, out var ex) + ? ex : new PuzzleMissionEntry { Id = s.Id }; + + entry.MissionName = s.MissionName; + entry.AchievedMessage = s.AchievedMessage; + entry.RequireNumber = s.RequireNumber; + entry.CampaignCommenceTime = s.CampaignCommenceTime; + entry.OrderId = s.OrderId; + entry.RewardType = s.RewardType; + entry.RewardDetailId = s.RewardDetailId; + entry.RewardNumber = s.RewardNumber; + entry.TargetPuzzleGroupId = s.TargetPuzzleGroupId; + + if (ex is null) + { + context.PuzzleMissions.Add(entry); + existing[s.Id] = entry; + created++; + } + else updated++; + } + + await context.SaveChangesAsync(); + Console.WriteLine($"[PuzzleImporter] Missions +{created}/~{updated}"); + return created + updated; + } +} diff --git a/SVSim.Bootstrap/Models/Seed/PuzzleGroupSeed.cs b/SVSim.Bootstrap/Models/Seed/PuzzleGroupSeed.cs new file mode 100644 index 0000000..7f64c68 --- /dev/null +++ b/SVSim.Bootstrap/Models/Seed/PuzzleGroupSeed.cs @@ -0,0 +1,14 @@ +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace SVSim.Bootstrap.Models.Seed; + +public sealed class PuzzleGroupSeed +{ + [JsonPropertyName("id")] public int Id { get; set; } + [JsonPropertyName("basic_title_text_id")] public string BasicTitleTextId { get; set; } = ""; + [JsonPropertyName("puzzle_chara_id")] public int PuzzleCharaId { get; set; } + [JsonPropertyName("chara_id")] public int CharaId { get; set; } + [JsonPropertyName("sort_type")] public int SortType { get; set; } + [JsonPropertyName("difficulty_name_list")] public JsonElement DifficultyNameList { get; set; } +} diff --git a/SVSim.Bootstrap/Models/Seed/PuzzleMissionSeed.cs b/SVSim.Bootstrap/Models/Seed/PuzzleMissionSeed.cs new file mode 100644 index 0000000..69eb401 --- /dev/null +++ b/SVSim.Bootstrap/Models/Seed/PuzzleMissionSeed.cs @@ -0,0 +1,17 @@ +using System.Text.Json.Serialization; + +namespace SVSim.Bootstrap.Models.Seed; + +public sealed class PuzzleMissionSeed +{ + [JsonPropertyName("id")] public int Id { get; set; } + [JsonPropertyName("mission_name")] public string MissionName { get; set; } = ""; + [JsonPropertyName("achieved_message")] public string AchievedMessage { get; set; } = ""; + [JsonPropertyName("require_number")] public int RequireNumber { get; set; } + [JsonPropertyName("campaign_commence_time")] public long CampaignCommenceTime { get; set; } + [JsonPropertyName("order_id")] public int OrderId { get; set; } + [JsonPropertyName("reward_type")] public int RewardType { get; set; } + [JsonPropertyName("reward_detail_id")] public long RewardDetailId { get; set; } + [JsonPropertyName("reward_number")] public int RewardNumber { get; set; } + [JsonPropertyName("target_puzzle_group_id")] public int? TargetPuzzleGroupId { get; set; } +} diff --git a/SVSim.Bootstrap/Models/Seed/PuzzleSeed.cs b/SVSim.Bootstrap/Models/Seed/PuzzleSeed.cs new file mode 100644 index 0000000..9fdc965 --- /dev/null +++ b/SVSim.Bootstrap/Models/Seed/PuzzleSeed.cs @@ -0,0 +1,13 @@ +using System.Text.Json.Serialization; + +namespace SVSim.Bootstrap.Models.Seed; + +public sealed class PuzzleSeed +{ + [JsonPropertyName("id")] public int Id { get; set; } + [JsonPropertyName("group_id")] public int GroupId { get; set; } + [JsonPropertyName("puzzle_difficulty")] public int PuzzleDifficulty { get; set; } + [JsonPropertyName("is_additional")] public bool IsAdditional { get; set; } + [JsonPropertyName("is_playable")] public bool IsPlayable { get; set; } + [JsonPropertyName("release_condition_text_id")] public string ReleaseConditionTextId { get; set; } = ""; +} diff --git a/SVSim.Bootstrap/Program.cs b/SVSim.Bootstrap/Program.cs index 3d27476..51ec14f 100644 --- a/SVSim.Bootstrap/Program.cs +++ b/SVSim.Bootstrap/Program.cs @@ -78,6 +78,10 @@ public static class Program await new GlobalsImporter().ImportAllAsync(context, opts.CapturesDir); await new PracticeOpponentImporter().ImportAsync(context, opts.SeedDir); await new PaymentItemImporter().ImportAsync(context, opts.SeedDir); + var puzzleImporter = new PuzzleImporter(); + await puzzleImporter.ImportGroupsAsync(context, opts.SeedDir); + await puzzleImporter.ImportPuzzlesAsync(context, opts.SeedDir); + await puzzleImporter.ImportMissionsAsync(context, opts.SeedDir); // BuildDeck pipeline: series CSV → catalog JSON → package CSV. Catalog must run after // series CSV (FK on products → series) and before package CSV (so the catalog-side diff --git a/SVSim.UnitTests/Importers/PuzzleImporterTests.cs b/SVSim.UnitTests/Importers/PuzzleImporterTests.cs new file mode 100644 index 0000000..9dd53eb --- /dev/null +++ b/SVSim.UnitTests/Importers/PuzzleImporterTests.cs @@ -0,0 +1,154 @@ +using Microsoft.EntityFrameworkCore; +using Microsoft.Extensions.DependencyInjection; +using SVSim.Bootstrap.Importers; +using SVSim.Database; +using SVSim.Database.Models; +using SVSim.UnitTests.Infrastructure; + +namespace SVSim.UnitTests.Importers; + +public class PuzzleImporterTests +{ + private static string SeedDir => Path.Combine(AppContext.BaseDirectory, "Data", "seeds"); + + [Test] + public async Task ImportsGroups_PuzzlesAndMissions_from_seed_files() + { + using var factory = new SVSimTestFactory(); + using var scope = factory.Services.CreateScope(); + var db = scope.ServiceProvider.GetRequiredService(); + + var importer = new PuzzleImporter(); + await importer.ImportGroupsAsync(db, SeedDir); + await importer.ImportPuzzlesAsync(db, SeedDir); + await importer.ImportMissionsAsync(db, SeedDir); + + int groupCount = await db.PuzzleGroups.CountAsync(); + int puzzleCount = await db.Puzzles.CountAsync(); + int missionCount = await db.PuzzleMissions.CountAsync(); + + Assert.That(groupCount, Is.GreaterThan(0), "seed must contain groups"); + Assert.That(puzzleCount, Is.GreaterThan(0), "seed must contain puzzles"); + Assert.That(missionCount, Is.GreaterThan(0), "seed must contain missions"); + + // Every puzzle's GroupId must reference an existing group (FK satisfied). + var groupIds = await db.PuzzleGroups.Select(g => g.Id).ToListAsync(); + var groupIdSet = new HashSet(groupIds); + var puzzleGroupIds = await db.Puzzles.Select(p => p.GroupId).Distinct().ToListAsync(); + foreach (var gid in puzzleGroupIds) + { + Assert.That(groupIdSet, Does.Contain(gid), + $"puzzle references unknown group_id={gid}"); + } + } + + [Test] + public async Task Is_idempotent_on_rerun() + { + using var factory = new SVSimTestFactory(); + using var scope = factory.Services.CreateScope(); + var db = scope.ServiceProvider.GetRequiredService(); + + var importer = new PuzzleImporter(); + await importer.ImportGroupsAsync(db, SeedDir); + await importer.ImportPuzzlesAsync(db, SeedDir); + await importer.ImportMissionsAsync(db, SeedDir); + + int g1 = await db.PuzzleGroups.CountAsync(); + int p1 = await db.Puzzles.CountAsync(); + int m1 = await db.PuzzleMissions.CountAsync(); + + await importer.ImportGroupsAsync(db, SeedDir); + await importer.ImportPuzzlesAsync(db, SeedDir); + await importer.ImportMissionsAsync(db, SeedDir); + + Assert.That(await db.PuzzleGroups.CountAsync(), Is.EqualTo(g1)); + Assert.That(await db.Puzzles.CountAsync(), Is.EqualTo(p1)); + Assert.That(await db.PuzzleMissions.CountAsync(), Is.EqualTo(m1)); + } + + [Test] + public async Task Leaves_existing_rows_untouched_when_missing_from_seed() + { + using var factory = new SVSimTestFactory(); + using var scope = factory.Services.CreateScope(); + var db = scope.ServiceProvider.GetRequiredService(); + + const int legacyGroupId = 99999; + const int legacyPuzzleId = 99998; + const int legacyMissionId = 99997; + + db.PuzzleGroups.Add(new PuzzleGroupEntry + { + Id = legacyGroupId, + BasicTitleTextId = "legacy_group", + DifficultyNameListJson = "{\"legacy\":\"1\"}", + }); + db.Puzzles.Add(new PuzzleEntry + { + Id = legacyPuzzleId, + GroupId = legacyGroupId, + PuzzleDifficulty = 5, + ReleaseConditionTextId = "legacy_puzzle", + }); + db.PuzzleMissions.Add(new PuzzleMissionEntry + { + Id = legacyMissionId, + MissionName = "legacy_mission", + AchievedMessage = "legacy_achieved", + RequireNumber = 42, + }); + await db.SaveChangesAsync(); + + var importer = new PuzzleImporter(); + await importer.ImportGroupsAsync(db, SeedDir); + await importer.ImportPuzzlesAsync(db, SeedDir); + await importer.ImportMissionsAsync(db, SeedDir); + + var g = await db.PuzzleGroups.FindAsync(legacyGroupId); + Assert.That(g, Is.Not.Null); + Assert.That(g!.BasicTitleTextId, Is.EqualTo("legacy_group")); + + var p = await db.Puzzles.FindAsync(legacyPuzzleId); + Assert.That(p, Is.Not.Null); + Assert.That(p!.PuzzleDifficulty, Is.EqualTo(5)); + + var m = await db.PuzzleMissions.FindAsync(legacyMissionId); + Assert.That(m, Is.Not.Null); + Assert.That(m!.MissionName, Is.EqualTo("legacy_mission")); + Assert.That(m.RequireNumber, Is.EqualTo(42)); + } + + [Test] + public async Task Skips_rows_with_zero_id() + { + using var factory = new SVSimTestFactory(); + using var scope = factory.Services.CreateScope(); + var db = scope.ServiceProvider.GetRequiredService(); + + string tmp = Path.Combine(Path.GetTempPath(), $"seed-{Guid.NewGuid()}"); + Directory.CreateDirectory(tmp); + try + { + File.WriteAllText(Path.Combine(tmp, "puzzle-groups.json"), + "[{\"id\":0,\"basic_title_text_id\":\"junk\"}]"); + File.WriteAllText(Path.Combine(tmp, "puzzles.json"), + "[{\"id\":0,\"group_id\":1,\"puzzle_difficulty\":1}]"); + File.WriteAllText(Path.Combine(tmp, "puzzle-missions.json"), + "[{\"id\":0,\"mission_name\":\"junk\"}]"); + + var importer = new PuzzleImporter(); + await importer.ImportGroupsAsync(db, tmp); + await importer.ImportPuzzlesAsync(db, tmp); + await importer.ImportMissionsAsync(db, tmp); + + Assert.That(await db.PuzzleGroups.CountAsync(), Is.EqualTo(0), + "rows with id=0 must not be inserted into groups"); + Assert.That(await db.Puzzles.CountAsync(), Is.EqualTo(0), + "rows with id=0 must not be inserted into puzzles"); + Assert.That(await db.PuzzleMissions.CountAsync(), Is.EqualTo(0), + "rows with id=0 must not be inserted into missions"); + } + finally { Directory.Delete(tmp, true); } + } +} diff --git a/SVSim.UnitTests/Infrastructure/SVSimTestFactory.cs b/SVSim.UnitTests/Infrastructure/SVSimTestFactory.cs index d1ab71e..e6a895e 100644 --- a/SVSim.UnitTests/Infrastructure/SVSimTestFactory.cs +++ b/SVSim.UnitTests/Infrastructure/SVSimTestFactory.cs @@ -194,6 +194,10 @@ internal sealed class SVSimTestFactory : WebApplicationFactory // practice-opponent rows after the corresponding block was lifted out of GlobalsImporter. await new PracticeOpponentImporter().ImportAsync(ctx, seedDir); await new PaymentItemImporter().ImportAsync(ctx, seedDir); + var puzzleImporter = new PuzzleImporter(); + await puzzleImporter.ImportGroupsAsync(ctx, seedDir); + await puzzleImporter.ImportPuzzlesAsync(ctx, seedDir); + await puzzleImporter.ImportMissionsAsync(ctx, seedDir); } /// Convenience: bake the X-Test-Viewer-Id header into a fresh client.