Пример #1
0
        private void DownloadTitle(string tid)
        {
            // Run everything in a separate thread
            _ = Task.Run(() =>
            {
                IsIdle = false;

                // Search for the title in the collection
                Title title = Titles.First(o => o.TitleId == tid);

                // Set the window titles prop of the file which will be downloaded
                WindowTitle = $"[{title.Region}] [{title.TitleType}] {(String.IsNullOrEmpty(title.Name) ? title.TitleId : title.Name)}";

                AddLogEntry("Preparing download...");

                string baseUrl = SettingsProvider.Settings.NintendoBaseUrl + title.TitleId + "/";

                string saveDir = SettingsProvider.Settings.SavePath;

                if (String.IsNullOrEmpty(title.Name))
                {
                    saveDir = @$ "{saveDir}{title.TitleType}\\{title.TitleId.Replace(" ", " ")}";
                }
                else
                {
                    saveDir = @$ "{saveDir}{title.TitleType}\\[{title.Region}] {title.NameSanitized}";
                }

                // Create the save directory
                if (!Directory.Exists(saveDir))
                {
                    Directory.CreateDirectory(saveDir);
                }

                UInt64 titleSize = 0;
                Ticket ticket;
                Tmd tmd;

                try
                {
                    using (WebClient wc = new WebClient())
                    {
                        try
                        {
                            AddLogEntry(" + Downloading TMD from Nintendo CDN...");
                            tmd = new Tmd(wc.DownloadData(baseUrl + "tmd"));

                            AddLogEntry(" + Saving TMD - title.tmd");
                            File.WriteAllBytes(saveDir + "\\title.tmd", tmd.ExportTmdData());
                        }
                        catch (WebException e)
                        {
                            AddLogEntry(" + ERROR! Could not download TMD!");
                            Debug.WriteLine(e.Message);
                            throw e;
                        }
                        catch (IOException e)
                        {
                            AddLogEntry(" + ERROR! Could not save title.tmd!");
                            Debug.WriteLine(e.Message);
                            throw e;
                        }

                        for (int i = 0; i < tmd.GetContentCount(); i++)
                        {
                            titleSize += tmd.GetContentSize((uint)i);
                        }

                        AddLogEntry($" + Estimated Content Size: {FormatBytes(titleSize)}");

                        WindowTitle += " " + FormatBytes(titleSize);

                        string titleType = title.TitleType;;

                        if (!(titleType.Contains("game", StringComparison.OrdinalIgnoreCase) || titleType.Contains("demo", StringComparison.OrdinalIgnoreCase) || titleType.Contains("dlc", StringComparison.OrdinalIgnoreCase)))
                        {
                            AddLogEntry(" + Downloading Ticket from Nintendo CDN...");
                            try
                            {
                                byte[] cetk = wc.DownloadData(baseUrl + "cetk");
                                byte[] tik  = new byte[0x350];

                                for (int i = 0; i < tik.Length; i++)
                                {
                                    tik[i] = cetk[i];
                                }

                                ticket = new Ticket(tik);

                                AddLogEntry(" + Saving Ticket: title.tik");
                                try { File.WriteAllBytes(saveDir + "\\title.tik", ticket.ExportTicketData()); }
                                catch (IOException e) { AddLogEntry(" + ERROR! Could not save title.tik!"); Debug.WriteLine(e.Message); throw e; }
                            }
                            catch (WebException e)
                            {
                                AddLogEntry(" + ERROR! Could not download Ticket from Nintendo CDN!");
                                Debug.WriteLine(e.Message);
                                throw e;
                            }
                        }
                        else
                        {
                            AddLogEntry(" + Generating Fake Ticket...");
                            ticket = new Ticket();
                            ticket.PatchTitleID(title.TitleId);
                            ticket.PatchTitleKey(title.TitleKey);
                            ticket.PatchTitleVersion(tmd.GetTitleVersion());

                            if (titleType.Contains("dlc", StringComparison.OrdinalIgnoreCase))
                            {
                                AddLogEntry(" + Patching Ticket: Applying \"DLC Unlock All\" Patch");
                                ticket.PatchDLCUnlockAll();
                            }

                            if (titleType.Contains("demo", StringComparison.OrdinalIgnoreCase))
                            {
                                AddLogEntry(" + Patching Ticket: Applying \"Demo Remove Time Limit\" Patch");
                                ticket.PatchDemoKillTimeLimit();
                            }

                            AddLogEntry(" + Saving Ticket: title.tik");
                            try { File.WriteAllBytes(saveDir + "\\title.tik", ticket.ExportTicketData()); }
                            catch (IOException e) { AddLogEntry(" + ERROR! Could not save title.tik!"); Debug.WriteLine(e.Message); throw e; }
                        }

                        AddLogEntry(" + Saving title.cert...");
                        try { File.WriteAllBytes(saveDir + "\\title.cert", Utils.GetByteArrayFromHexString("00010003704138EFBBBDA16A987DD901326D1C9459484C88A2861B91A312587AE70EF6237EC50E1032DC39DDE89A96A8E859D76A98A6E7E36A0CFE352CA893058234FF833FCB3B03811E9F0DC0D9A52F8045B4B2F9411B67A51C44B5EF8CE77BD6D56BA75734A1856DE6D4BED6D3A242C7C8791B3422375E5C779ABF072F7695EFA0F75BCB83789FC30E3FE4CC8392207840638949C7F688565F649B74D63D8D58FFADDA571E9554426B1318FC468983D4C8A5628B06B6FC5D507C13E7A18AC1511EB6D62EA5448F83501447A9AFB3ECC2903C9DD52F922AC9ACDBEF58C6021848D96E208732D3D1D9D9EA440D91621C7A99DB8843C59C1F2E2C7D9B577D512C166D6F7E1AAD4A774A37447E78FE2021E14A95D112A068ADA019F463C7A55685AABB6888B9246483D18B9C806F474918331782344A4B8531334B26303263D9D2EB4F4BB99602B352F6AE4046C69A5E7E8E4A18EF9BC0A2DED61310417012FD824CC116CFB7C4C1F7EC7177A17446CBDE96F3EDD88FCD052F0B888A45FDAF2B631354F40D16E5FA9C2C4EDA98E798D15E6046DC5363F3096B2C607A9D8DD55B1502A6AC7D3CC8D8C575998E7D796910C804C495235057E91ECD2637C9C1845151AC6B9A0490AE3EC6F47740A0DB0BA36D075956CEE7354EA3E9A4F2720B26550C7D394324BC0CB7E9317D8A8661F42191FF10B08256CE3FD25B745E5194906B4D61CB4C2E000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000526F6F7400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001434130303030303030330000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000007BE8EF6CB279C9E2EEE121C6EAF44FF639F88F078B4B77ED9F9560B0358281B50E55AB721115A177703C7A30FE3AE9EF1C60BC1D974676B23A68CC04B198525BC968F11DE2DB50E4D9E7F071E562DAE2092233E9D363F61DD7C19FF3A4A91E8F6553D471DD7B84B9F1B8CE7335F0F5540563A1EAB83963E09BE901011F99546361287020E9CC0DAB487F140D6626A1836D27111F2068DE4772149151CF69C61BA60EF9D949A0F71F5499F2D39AD28C7005348293C431FFBD33F6BCA60DC7195EA2BCC56D200BAF6D06D09C41DB8DE9C720154CA4832B69C08C69CD3B073A0063602F462D338061A5EA6C915CD5623579C3EB64CE44EF586D14BAAA8834019B3EEBEED3790001000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000100042EA66C66CFF335797D0497B77A197F9FE51AB5A41375DC73FD9E0B10669B1B9A5B7E8AB28F01B67B6254C14AA1331418F25BA549004C378DD72F0CE63B1F7091AAFE3809B7AC6C2876A61D60516C43A63729162D280BE21BE8E2FE057D8EB6E204242245731AB6FEE30E5335373EEBA970D531BBA2CB222D9684387D5F2A1BF75200CE0656E390CE19135B59E14F0FA5C1281A7386CCD1C8EC3FAD70FBCE74DEEE1FD05F46330B51F9B79E1DDBF4E33F14889D05282924C5F5DC2766EF0627D7EEDC736E67C2E5B93834668072216D1C78B823A072D34FF3ECF9BD11A29AF16C33BD09AFB2D74D534E027C19240D595A68EBB305ACC44AB38AB820C6D426560C000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000526F6F742D43413030303030303033000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000143503030303030303062000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000137A080BA689C590FD0B2F0D4F56B632FB934ED0739517B33A79DE040EE92DC31D37C7F73BF04BD3E44E20AB5A6FEAF5984CC1F6062E9A9FE56C3285DC6F25DDD5D0BF9FE2EFE835DF2634ED937FAB0214D104809CF74B860E6B0483F4CD2DAB2A9602BC56F0D6BD946AED6E0BE4F08F26686BD09EF7DB325F82B18F6AF2ED525BFD828B653FEE6ECE400D5A48FFE22D538BB5335B4153342D4335ACF590D0D30AE2043C7F5AD214FC9C0FE6FA40A5C86506CA6369BCEE44A32D9E695CF00B4FD79ADB568D149C2028A14C9D71B850CA365B37F70B657791FC5D728C4E18FD22557C4062D74771533C70179D3DAE8F92B117E45CB332F3B3C2A22E705CFEC66F6DA3772B000100010000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010004919EBE464AD0F552CD1B72E7884910CF55A9F02E50789641D896683DC005BD0AEA87079D8AC284C675065F74C8BF37C88044409502A022980BB8AD48383F6D28A79DE39626CCB2B22A0F19E41032F094B39FF0133146DEC8F6C1A9D55CD28D9E1C47B3D11F4F5426C2C780135A2775D3CA679BC7E834F0E0FB58E68860A71330FC95791793C8FBA935A7A6908F229DEE2A0CA6B9B23B12D495A6FE19D0D72648216878605A66538DBF376899905D3445FC5C727A0E13E0E2C8971C9CFA6C60678875732A4E75523D2F562F12AABD1573BF06C94054AEFA81A71417AF9A4A066D0FFC5AD64BAB28B1FF60661F4437D49E1E0D9412EB4BCACF4CFD6A3408847982000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000526F6F742D43413030303030303033000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000158533030303030303063000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000137A0894AD505BB6C67E2E5BDD6A3BEC43D910C772E9CC290DA58588B77DCC11680BB3E29F4EABBB26E98C2601985C041BB14378E689181AAD770568E928A2B98167EE3E10D072BEEF1FA22FA2AA3E13F11E1836A92A4281EF70AAF4E462998221C6FBB9BDD017E6AC590494E9CEA9859CEB2D2A4C1766F2C33912C58F14A803E36FCCDCCCDC13FD7AE77C7A78D997E6ACC35557E0D3E9EB64B43C92F4C50D67A602DEB391B06661CD32880BD64912AF1CBCB7162A06F02565D3B0ECE4FCECDDAE8A4934DB8EE67F3017986221155D131C6C3F09AB1945C206AC70C942B36F49A1183BCD78B6E4B47C6C5CAC0F8D62F897C6953DD12F28B70C5B7DF751819A98346526250001000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000")); }
                        catch (IOException e) { AddLogEntry(" + ERROR! Could not save title.cert!"); Debug.WriteLine(e.Message); throw e; }
                    }

                    DownloadFiles(baseUrl, saveDir, tmd);

                    AddLogEntry(String.Format(" + {0}: {1} downloaded!", title.TitleType, Path.GetFileName(saveDir)));

                    Decrypt(title, saveDir);
                }
                catch (Exception)
                {
                    AddLogEntry(" + ABORTING DOWNLOAD OF THIS TITLE");
                }
                finally
                {
                    AddLogEntry(String.Format(" + -------------------------------------------------------------------", Path.GetFileName(saveDir)));
                    IsIdle = true;
                }
            });
        }
Пример #2
0
        private void DownloadFiles(string baseUrl, string saveDir, Tmd tmd)
        {
            List <string> contentIds = new List <string>();

            // Download Content .app and .h3 files
            for (uint i = 0; i < tmd.GetContentCount(); i++)
            {
                contentIds.Add(tmd.GetContentIDString(i));
            }

            Parallel.ForEach(contentIds, new ParallelOptions {
                MaxDegreeOfParallelism = SettingsProvider.Settings.MaxDegreeOfParallelism
            }, (content, state, i) =>
            {
                using WebClient wc = new WebClient();
                string cidStr      = content;

                // .app files
                try
                {
                    string contentFilePath = saveDir + "\\" + cidStr + ".app";

                    if (SettingsProvider.Settings.SkipExistingFiles && File.Exists(contentFilePath) && (ulong)contentFilePath.GetFileLength() == tmd.GetContentSize((uint)i))
                    {
                        AddLogEntry(" + File: " + cidStr + ".app exists with correct size, skipping download...");
                    }
                    else
                    {
                        int pad = contentIds.Count.ToString().Length;
                        AddLogEntry(String.Format(" + Downloading Content No. {0} / {1} - {2}.app ({3} bytes)",
                                                  (i + 1).ToString().PadLeft(pad),
                                                  contentIds.Count.ToString().PadLeft(pad),
                                                  cidStr,
                                                  string.Format("{0:N0}", tmd.GetContentSize((uint)i))));
                        wc.DownloadFile(baseUrl + cidStr, saveDir + "\\" + cidStr + ".app");
                    }
                }
                catch (WebException e)
                {
                    AddLogEntry(" + ERROR! Could not download " + cidStr + ".app");
                    throw e;
                }
                catch (IOException e)
                {
                    AddLogEntry(" + ERROR! Could not save " + cidStr + ".app");
                    throw e;
                }

                if (SettingsProvider.Settings.DownloadH3Files)
                {
                    // .h3 files
                    try
                    {
                        AddLogEntry(String.Format(" + Downloading H3 for Content No.{0} from Nintendo CDN - {1}.h3", i + 1, cidStr));
                        wc.DownloadFile(baseUrl + cidStr + ".h3", saveDir + "\\" + cidStr + ".h3");
                    }
                    catch (WebException e)
                    {
                        if (((HttpWebResponse)e.Response).StatusCode == HttpStatusCode.NotFound)
                        {
                            AddLogEntry(String.Format(" + File {0}.h3 not found, ignoring...", cidStr));
                        }
                        else
                        {
                            AddLogEntry(" + ERROR! Could not download " + cidStr + ".h3");
                            throw e;
                        }
                    }
                    catch (IOException e)
                    {
                        AddLogEntry(" + ERROR! Could not save " + cidStr + ".h3");
                        throw e;
                    }
                }
            });
        }