Unpacking Software Livestream

Join our monthly Unpacking Software livestream to hear about the latest news, chat and opinion on packaging, software deployment and lifecycle management!

Learn More

Chocolatey Product Spotlight

Join the Chocolatey Team on our regular monthly stream where we put a spotlight on the most recent Chocolatey product releases. You'll have a chance to have your questions answered in a live Ask Me Anything format.

Learn More

Chocolatey Coding Livestream

Join us for the Chocolatey Coding Livestream, where members of our team dive into the heart of open source development by coding live on various Chocolatey projects. Tune in to witness real-time coding, ask questions, and gain insights into the world of package management. Don't miss this opportunity to engage with our team and contribute to the future of Chocolatey!

Learn More

Calling All Chocolatiers! Whipping Up Windows Automation with Chocolatey Central Management

Webinar from
Wednesday, 17 January 2024

We are delighted to announce the release of Chocolatey Central Management v0.12.0, featuring seamless Deployment Plan creation, time-saving duplications, insightful Group Details, an upgraded Dashboard, bug fixes, user interface polishing, and refined documentation. As an added bonus we'll have members of our Solutions Engineering team on-hand to dive into some interesting ways you can leverage the new features available!

Watch On-Demand
Chocolatey Community Coffee Break

Join the Chocolatey Team as we discuss all things Community, what we do, how you can get involved and answer your Chocolatey questions.

Watch The Replays
Chocolatey and Intune Overview

Webinar Replay from
Wednesday, 30 March 2022

At Chocolatey Software we strive for simple, and teaching others. Let us teach you just how simple it could be to keep your 3rd party applications updated across your devices, all with Intune!

Watch On-Demand
Chocolatey For Business. In Azure. In One Click.

Livestream from
Thursday, 9 June 2022

Join James and Josh to show you how you can get the Chocolatey For Business recommended infrastructure and workflow, created, in Azure, in around 20 minutes.

Watch On-Demand
The Future of Chocolatey CLI

Livestream from
Thursday, 04 August 2022

Join Paul and Gary to hear more about the plans for the Chocolatey CLI in the not so distant future. We'll talk about some cool new features, long term asks from Customers and Community and how you can get involved!

Watch On-Demand
Hacktoberfest Tuesdays 2022

Livestreams from
October 2022

For Hacktoberfest, Chocolatey ran a livestream every Tuesday! Re-watch Cory, James, Gary, and Rain as they share knowledge on how to contribute to open-source projects such as Chocolatey CLI.

Watch On-Demand

Downloads:

584

Downloads of v 2.6.4:

142

Last Update:

13 Jan 2025

Package Maintainer(s):

Software Author(s):

  • Deco

Tags:

dlss nvidia gaming update-tool

DLSS Updater

This is not the latest version of DLSS Updater available.

  • 1
  • 2
  • 3

2.6.4 | Updated: 13 Jan 2025

Downloads:

584

Downloads of v 2.6.4:

142

Maintainer(s):

Software Author(s):

  • Deco

DLSS Updater 2.6.4

This is not the latest version of DLSS Updater available.

Legal Disclaimer: Neither this package nor Chocolatey Software, Inc. are affiliated with or endorsed by Deco. The inclusion of Deco trademark(s), if any, upon this webpage is solely to identify Deco goods or services and not for commercial purposes.

  • 1
  • 2
  • 3

Some Checks Have Failed or Are Not Yet Complete

Not All Tests Have Passed


Validation Testing Passed


Verification Testing Passed

Details

Scan Testing Resulted in Flagged as a Note:

At least one file within this package has greater than 0 detections, but less than 5

Details
Learn More

Deployment Method: Individual Install, Upgrade, & Uninstall

To install DLSS Updater, run the following command from the command line or from PowerShell:

>

To upgrade DLSS Updater, run the following command from the command line or from PowerShell:

>

To uninstall DLSS Updater, run the following command from the command line or from PowerShell:

>

Deployment Method:

NOTE

This applies to both open source and commercial editions of Chocolatey.

1. Enter Your Internal Repository Url

(this should look similar to https://community.chocolatey.org/api/v2/)


2. Setup Your Environment

1. Ensure you are set for organizational deployment

Please see the organizational deployment guide

2. Get the package into your environment

  • Open Source or Commercial:
    • Proxy Repository - Create a proxy nuget repository on Nexus, Artifactory Pro, or a proxy Chocolatey repository on ProGet. Point your upstream to https://community.chocolatey.org/api/v2/. Packages cache on first access automatically. Make sure your choco clients are using your proxy repository as a source and NOT the default community repository. See source command for more information.
    • You can also just download the package and push it to a repository Download

3. Copy Your Script

choco upgrade dlss-updater -y --source="'INTERNAL REPO URL'" --version="'2.6.4'" [other options]

See options you can pass to upgrade.

See best practices for scripting.

Add this to a PowerShell script or use a Batch script with tools and in places where you are calling directly to Chocolatey. If you are integrating, keep in mind enhanced exit codes.

If you do use a PowerShell script, use the following to ensure bad exit codes are shown as failures:


choco upgrade dlss-updater -y --source="'INTERNAL REPO URL'" --version="'2.6.4'" 
$exitCode = $LASTEXITCODE

Write-Verbose "Exit code was $exitCode"
$validExitCodes = @(0, 1605, 1614, 1641, 3010)
if ($validExitCodes -contains $exitCode) {
  Exit 0
}

Exit $exitCode

- name: Install dlss-updater
  win_chocolatey:
    name: dlss-updater
    version: '2.6.4'
    source: INTERNAL REPO URL
    state: present

See docs at https://docs.ansible.com/ansible/latest/modules/win_chocolatey_module.html.


chocolatey_package 'dlss-updater' do
  action    :install
  source   'INTERNAL REPO URL'
  version  '2.6.4'
end

See docs at https://docs.chef.io/resource_chocolatey_package.html.


cChocoPackageInstaller dlss-updater
{
    Name     = "dlss-updater"
    Version  = "2.6.4"
    Source   = "INTERNAL REPO URL"
}

Requires cChoco DSC Resource. See docs at https://github.com/chocolatey/cChoco.


package { 'dlss-updater':
  ensure   => '2.6.4',
  provider => 'chocolatey',
  source   => 'INTERNAL REPO URL',
}

Requires Puppet Chocolatey Provider module. See docs at https://forge.puppet.com/puppetlabs/chocolatey.


4. If applicable - Chocolatey configuration/installation

See infrastructure management matrix for Chocolatey configuration elements and examples.

Package Approved

This package was approved by moderator Windos on 14 Jan 2025.

Description

DLSS Updater is a utility that automatically updates DLSS (Deep Learning Super Sampling) DLLs for games across multiple platforms including Steam, EA Play, Ubisoft, Epic Games, GOG, and Battle.net. It simplifies the process of keeping your games up-to-date with the latest DLSS improvements. Note that this package does not contain malware, it is a limitation of Pyinstaller with the binary not being signed, please see https://github.com/pyinstaller/pyinstaller/issues/6754#issuecomment-1100821249 for more details.


tools\DLSS_Updater.exe
md5: C5319AA258A522960B0B74EC0A4A31CE | sha1: D41DC45CF655CA29670BFC7EE77C1B7DDD39DF4B | sha256: 8A71B215DD3BAE29C4C5A40F13054FFF91AA1DE52C5886B3D86BDA720D91E0E8 | sha512: 29E9DEAB90D58D58EAC7206B5C7C234EDE419718CA944F22C0AA6CCA0D6A8E570540EF6F92A5E98958A967B9300FBABCA1E49C8B8978F9B7A4E51DA836721532
tools\dlss_updater.png
 
tools\VERIFICATION.TXT
VERIFICATION
Verification is intended to assist the Chocolatey moderators and community
in verifying that this package's contents are trustworthy.

This package is published by the DLSS Updater Project itself. The binaries are 
identical to those published in the GitHub releases section.

Project URL: https://github.com/Recol/DLSS-Updater
Releases: https://github.com/Recol/DLSS-Updater/releases

To verify the binaries:

1. Download the official release from the GitHub releases page
2. Generate a checksum using Get-FileHash in PowerShell
   Get-FileHash DLSS_Updater.exe -Algorithm SHA256
3. Compare the checksum with the one provided in the release notes or in this file

Alternatively, you can use the verification methods provided by GitHub to ensure
the release you're downloading matches the source code at the tagged commit.

Note: This file will be updated with each release to include the specific
checksum for the current version. Always refer to the VERIFICATION.txt
included in the package for the most up-to-date verification information.

Checksum (SHA256) for DLSS_Updater.exe: 8A71B215DD3BAE29C4C5A40F13054FFF91AA1DE52C5886B3D86BDA720D91E0E8
tools\_internal\api-ms-win-core-console-l1-1-0.dll
md5: 854560AB49893FC0EB1C3D2ADE1E11FB | sha1: 5B131B6154D0C56ABFB5EAD12BBB5E82E3F2CCC8 | sha256: 79E4EDF3FF63A7B1B279DC6352594F4512E0789A3D5E80CD4A34A68129DF4161 | sha512: F641C944D05B849652715D95FEA2E4431056D0AABA00D28E7D1502C522E66799D1CF277C3442446D940F6FA1A285BB0F2999085D1DCC660BB730AFF37DE2BB79
tools\_internal\api-ms-win-core-datetime-l1-1-0.dll
md5: F51A025B8C1A2146847DF21CB0B4136A | sha1: 45DC4A50C0C2A9B32DDAC679D705E7501D95E8E1 | sha256: 9600F43772639BA115FC7F45B1EDD775B31BBCB7202FA87C78490383DAA7030C | sha512: 3CD225F2CE1B91B7DC4C27E144CC97A36F997C0A0259395EB9EA9F57471FBA589855B810B5D5326FDCCD5C9E9CF06D889B758E374D389CBDCBD89601B17DE545
tools\_internal\api-ms-win-core-debug-l1-1-0.dll
md5: E1DFEB517A691FD91247CFBE4349B41D | sha1: 5263D1F6D103DDF18A0590B41FFC582D61F4362B | sha256: 43533C66AE70709723E12BC80F047644D68B0282ABC76B4C952461EE8554C8E2 | sha512: F5271BC1B8BD387A46F0FF5103D4C468C0B458D2F1DC0CBACE0F7A568EBD0B9C8D0414D961118687CE1A7876E28D82ED531CCA95DD1661F208FDFDD4223FEADB
tools\_internal\api-ms-win-core-errorhandling-l1-1-0.dll
md5: B7EA2415828AB7E8234EE71CC1274312 | sha1: D053DF9B9CC701978D159E48A9F5422A275220C9 | sha256: BE358D7F9A80C56125C872D98469D470C962EB89A87FE7C3EEB2813AB691F162 | sha512: AF26B547F31080E359002B1A1FA71D76A2BD4771B1C5AA9584D8B0D64911D889A8AF8BD46D80FC36A9AD2F5E04881ED0A640C8AAB7F0A1D729B5032D84B98664
tools\_internal\api-ms-win-core-fibers-l1-1-0.dll
md5: 401B34BE80C11C38783E1DDB47799779 | sha1: 0AD8F38BBBD41CEB5CAA6E2B44D308FA4707CF1D | sha256: 772372F20239899FB25D1A72E0210D729A9AC9CCE8E036922592405BCDD9D287 | sha512: BC596988318D2877DBE52AAAC19470E61F441F61620EC6D72B8025B427D7772AFE802BFE1DFB83A29A8A9F1BF79F22812DD4688253037FB1C5D0139381AB92F9
tools\_internal\api-ms-win-core-fibers-l1-1-1.dll
md5: 8F12F2B949081422329527DE9F752C52 | sha1: E69A417535258F9D7CBB762171D76D218F58F6AA | sha256: 8FCAE6D9A2A43FAAFB9B78D22CCE9CD2B4589952A81F713CF26E6DCA0C198C6C | sha512: A985086B592363103B786E57E623945F316B34E10A34D12FA47E385DB0C999B8F143FC7D8DB19220A1FE2D7E1AA63FBEFC4052D95D7311357C6CA234CC360F42
tools\_internal\api-ms-win-core-file-l1-1-0.dll
md5: 13B7840BDD7312959FD2F134CAF81B04 | sha1: 85E9D1981596F8D8F1584F89FF7243B02CB91787 | sha256: 57A24B7D585BA98AB0DFF395C62525F10F498BF0BE4871ADC8C805B997D7368A | sha512: 2C9573413D842A0956F914AAEEF25280F6AFA145B30E79E40B1CAAA62B482C26438283AFC08BAD568D500AC98E009AA85290F0B9DB0C226829E9A8D9CA10617B
tools\_internal\api-ms-win-core-file-l1-2-0.dll
md5: D2F264B9F61BBBEA858CCA1F1A85FFF4 | sha1: 98903EA36BC421969360018EE953D5E293C8651E | sha256: 00AF59B43E70769D1CB516FF9A83A6E11D27D44889B18B498D10E2E5EB2846FA | sha512: A5B0056FCE6E6B40EA95FF5DF451C91864A963DB3A97781729C9816BA72C1BEA92EB1AC9AD7BE33F79FC9299CD10BFC5B074B7BDC0DD049F40019BBEDD1B3916
tools\_internal\api-ms-win-core-file-l2-1-0.dll
md5: 9C4AA976FBBA6EE469DB69F3268E0EE0 | sha1: FBC510424960D4D6CA8959AB8A79ED7E0106D894 | sha256: B8EE3713B2FAC086263084EB76C91906F1773EBE427ED012CC5AC77CFA506BFE | sha512: D134D8B876507616D7B6D97421014D61D04AD82785CBBF7F9DFB44D1E3361410B4590613A2D7D4F4683533FFAF099B88FDF505F1B8C578FD933394187DD17388
tools\_internal\api-ms-win-core-handle-l1-1-0.dll
md5: 2F9076385FCE7A50D921C2C04CE82357 | sha1: 19D86416DFB12B3FD03DBF5DD23ACBD7ABA39E98 | sha256: 0069D044789AE935144AB20AFA81947E523F7879E72781C6352060F182F16C22 | sha512: ACA0F1CEA0737967760D2E1D967CECB8CDDA36D6CC729064643CD662313C626CCE546210D6F12F653248AA5D6B9991A1BD64D0785257369CECAE3DE1AE67734E
tools\_internal\api-ms-win-core-heap-l1-1-0.dll
md5: 7893D219F6BDA4BEAFE1937FFC026386 | sha1: 6A80B511FEF91031F707266DD358AFD5D624737D | sha256: 95767FD45416F86B2A16BF50E971F39A9F64A680F6CDF1D6BDA9C64E633FFF6C | sha512: 24A3617E3B2FC2810D06A92813591E1C5F475C47648971D9D158120D96A9504D5EA3FB24F6AE2A9770A0034EFCA374FD1F1A8E0C944A32D201CB617BDA01B096
tools\_internal\api-ms-win-core-interlocked-l1-1-0.dll
md5: E8745CE7467509E4B59522DE48EEF43E | sha1: 9A1058A7124D87FD6EA02442C1BA5D68F86A86A6 | sha256: 6E65A8482E9867A16F9E6398335139500C6D5E2F56A232FDD33F7F46541488BE | sha512: 97F5451494B1969806C010DD552A79556960D9B095CB245DA83554A53B004885111CC39A53F0466EFDB0E5F1F69B2990D19CE126F529E5F79A0DD0AD2E7EE672
tools\_internal\api-ms-win-core-kernel32-legacy-l1-1-1.dll
md5: B4A768285A5F30DB0FEC2114714D4CE0 | sha1: FD6DFC23C36D09123AF87075C5130BA87E2FD81B | sha256: 569618315C6B659BC5FB0799A0A2480371425570E7F195395B5159BA12257EFA | sha512: 2B45ABCC9EDF1A712D9F5C291A992FA198472D679A66EEDD211DB22836051DAE1FEB6235FF839F4B7A3365D3B010EB6E7AEF369D4D404CF1B9043867923E4347
tools\_internal\api-ms-win-core-libraryloader-l1-1-0.dll
md5: 3205ABC6AFC72E7D9D78D6BB736068CB | sha1: F14C3809E15DC1A39BA4B815D8B2784C3B451464 | sha256: 6614E8C94F8D2E48417EE9EC2155DFC2D8DD7BD6B78C89617ACE90CB851114D2 | sha512: 1C9C61157D745A6948C941371F1C0CE3DB32CEFEA8F9FD5797628D6C461650F765C3EDEDE13F337F04C8317EA256AC06D7520EDBE9FBED1F777455B4CF0BE909
tools\_internal\api-ms-win-core-localization-l1-2-0.dll
md5: E1877632019BB32967C40767AFF863B6 | sha1: 2268935F0C872FEED067C3C17C70E5092CE301A1 | sha256: D1ECD2C21DF1D7B130BA0F1A1D99FB8866727BFFB3862883618A2CF545659DF8 | sha512: 98E620CE28B776DC6D2B39FA043B1E96555F641263E7254510587DFAD9EBFBDDEF0558756035657BA10C7B800B72A322589725FFCCCCC4EC5847D20E7A74023F
tools\_internal\api-ms-win-core-memory-l1-1-0.dll
md5: 740DD1CB6EC07DF5E43A2CBE3B66DD80 | sha1: E39493FD219C57F50D47119E94AEE7C8BBC3863B | sha256: 03A723BCBCC88604015B66C85589AFB5FD0A9F0E3F012160DCCA5F4AC0762B49 | sha512: 620F7743FE187B455C94177BE4FE133500F94566E79402F60B0ECC7AD11BF3350E6425839692E0C7C5A4F5159DF64C240314F1EDB7496BDE48B5590E43D0564D
tools\_internal\api-ms-win-core-namedpipe-l1-1-0.dll
md5: 9E4398814CE476F2554DC43EDB07DDBB | sha1: DB1C663354D219F9E95E4CA1B604CB77237716CD | sha256: 3C6ECC4E2284B19B8E0E673318FF9CC29F45041B2A0EA2705A8A8048D9276BA0 | sha512: BB7DB2BA4DC62B3497F36F2E58FF122665A76AB94AFFC6EE2E5E491052D8EBF389773CFCFBE262A4D00539EFABFA983BB68EEB8D70BD8E14F69093FE882BB81E
tools\_internal\api-ms-win-core-processenvironment-l1-1-0.dll
md5: 0055F1424D58A9AFE0D3362BC27DC2E3 | sha1: 568343A6830CDC9C74F9C0FC4743A35B086C53A5 | sha256: 541BAC07D88E28DDAEF0A0392EF3EBFD513A161D0923A9F361671C54F362D341 | sha512: 1D43913D9525D9B8C3C46DBAF57BF26EF251A377B000B4F3DF09226F6B529971069D4199B69206C6839925E4D02C5729C046C49A3D77E0E5165B6EDDD2AADE96
tools\_internal\api-ms-win-core-processthreads-l1-1-0.dll
md5: 17D9AC28553C5404D110BCDB6FAE4D90 | sha1: EA7B17476BE37D30EE2D7DCC818161FAE3157947 | sha256: E714FFDE0C79FAFE8067C86BCDF4EBB522AF00F741F655B074F46518557FF149 | sha512: 447660319AD9CB652C712C9ACDA67E9378F2D2DC5695A44DC24BC13E6B1359F97C86742FC3EA9649A9C0C9A105800E7F50F5C0D34984692DE95CA2D69E3D50CA
tools\_internal\api-ms-win-core-processthreads-l1-1-1.dll
md5: 774DE3D2577B4F6E50CC9CBE01069D03 | sha1: 8CBD24E84DCCB39630FD327744AB98DBA22489C9 | sha256: 94A70E7CD72242E29E0D0ECA78A2474AA1CC5CD529E29DCC62F680A61D47D6A6 | sha512: F89CE01766882082467EFC18BD9D236D7F3F56EE09E287D41413FF870A0B81135BEFD896F3B26FBC6D214BE795C0C06611B9B3BA9546FE1802C2C9E1FC5E27ED
tools\_internal\api-ms-win-core-profile-l1-1-0.dll
md5: CF1C69D66B674DA34801FA8791CCAA5F | sha1: EEA39A5948F576D319846606E8A23FDBD17D4547 | sha256: 9EAA28A9F953E852739B70703EF804F36DB33EE0CED4A37322F5DB656B1C42B9 | sha512: 4DCCB8582052C60ECF0D0BB3C70C0FB6963A4F9890DBB319010F10AE0C7D543EBCE6D8CE9B05FFDE38D1AB2ED37B2008ABF2812874724615CC02115B3A861AA7
tools\_internal\api-ms-win-core-rtlsupport-l1-1-0.dll
md5: 0986D5C7A8D89000C279B99843686783 | sha1: C7B5347C0B0A4800CAE0E2C37F96900213D60CA0 | sha256: EB8CE6EF361CB823257C9C837D046E7048C2C1FE52A25A12C5FDDB0034CF9FCE | sha512: 43A1F154AEB3C13D4C8C4C2E182BCCD7CAE8CB4643B86C480B2C9D9914E38057D13806BC406F38A00686CD0B8BE66BFF8FB4102AD1F728F079EDA998D57DCFB1
tools\_internal\api-ms-win-core-string-l1-1-0.dll
md5: 20291E55EB1C1866A3EAF252416DF69B | sha1: AA9D246B2EE7401BCB4746A71404EA0BF483029A | sha256: 634F1E2EEC066FE2F74A25F507CA37D1D979B982CF944975D1488E0435B86AD2 | sha512: 3406D33B48CA997A895D7A3EB6EF9DA8DBDC2B89D517409A9475B80EC95D18E274D4314A164CC306960CBBC848653D79F6E4B13425208E2B790CCCA5154236B6
tools\_internal\api-ms-win-core-synch-l1-1-0.dll
md5: BCE65FACD640D4B35ADB187DC1BE7180 | sha1: DFA96ADC02501F9CC0F88BA16441C47225477E34 | sha256: 50C78541FD07BC271B49259BF4D56E8885461371BF0852DD75E99E824BD4E754 | sha512: 8BF30C64D708835C246A44F5640805EA60D2577F472F6D0C56DDF66C10A33D8E0488E79B0A53C60BEFC5C0A583734220BF957FB66DD4D181320D8589D65A576E
tools\_internal\api-ms-win-core-synch-l1-2-0.dll
md5: 447AC6ACDE90CD2EE991885103E10742 | sha1: E674908B19BDC62EA02F3D53C2A7A5D05990D774 | sha256: 359C5D1221CFAB34B70D4F55E178CCD93F54A6DE3DA39C7472D67E7E330E300E | sha512: 59F429B5BC95F67B0192CC70209AA79B2001694778E84241582D9E3A20D065B087428ABA52EB4246E13755CE0D56DF20CE6FBA465CDE3F96D05ECBF486162B53
tools\_internal\api-ms-win-core-sysinfo-l1-1-0.dll
md5: ABCBE7F6DCCEEA80952092957F797A80 | sha1: 0543160EA20803E535ABC83064C86C1AEFB19556 | sha256: 154B5470E4E265AD29248A571686F6AAFCCEF9A8B2435A8633A70328C10BB371 | sha512: 2DF2A33C67BDB657F8BD1D4D01C6FFB93A82D998DDF034156DC57DB6CD45C72CAD6B1E18403C3EEF2FEC74F6094FB6EB6FF4E2643BAA03E122CB9C2F5EEBEEEC
tools\_internal\api-ms-win-core-sysinfo-l1-2-0.dll
md5: 7368E728C3648ED5A0F2582BD27EB583 | sha1: 4A92D740358468502D23BB18B463FC293D388D47 | sha256: 6D2A9C5745238EB4FC922652C72542703CFFF79A20DA2ABF18DC4A77618D75D7 | sha512: FC42442A3DEE5EBBCC4355C6498BBFEF9531CB427662DA929EF82A83A7667CA4B4976B0C688FC5D01DABEBFBDAD028C4D8EB04749DC9A6DE83D892A6377F9A6A
tools\_internal\api-ms-win-core-timezone-l1-1-0.dll
md5: EB7F8FE591DA1E09DE7594CC02772003 | sha1: 7D2F19EAAEA6CFDBC46BC302ED5ED2231CADC837 | sha256: F1061607D0DFE38C2585F22CEB326154C2CB1AC37761DF75E186F2D5D11B42B6 | sha512: 75B76CFE3899E45F9A0DCA67DC099531700CCEB06FD9CE209F641AF190C4CD6B96AF110E8C816E44BD194A8C109A1E85160A94DA633C49104A392EDB5BFE8E65
tools\_internal\api-ms-win-core-util-l1-1-0.dll
md5: D7B9B90357F4A2653ABB303BB78668DA | sha1: 52D4EAB0B938BB977EE8BF160844AC94C297FDC2 | sha256: AC72CD7713DA51EDF0B57616ED57326286A4F85DE8C1443EF60379DB9E3887F5 | sha512: BC034BED3A82F1A47207D02DDCE9F232F82110FB1A2B12EC1EBD468AF4A64E7DC547AA06ADADBD3993320FF5EDB0BE357CC7B4396160433E0ED5EDFF42D53B20
tools\_internal\api-ms-win-crt-conio-l1-1-0.dll
md5: 4B511048C52E62714D4AE8BF4D686DB7 | sha1: 2FFE23C3494E1930BB8BA9DC85BA7AFD50458A79 | sha256: 81CC477B01B8A2EB82E0EE21E9113604006A97281A95091823839335661569E6 | sha512: 328877E54FE0EF1A4A4AFFDEB11E19A064974ED28693C6DB9869025957F24AE2000FF39D1D1E03F804E2FB52323BBD3EF15BE4F99F0F1196A58F85FD4875A1EE
tools\_internal\api-ms-win-crt-convert-l1-1-0.dll
md5: BFBACEB27C68C53FD04488BB46AFE11E | sha1: 610F0CBDF9994B818FD518A99F559913370773D2 | sha256: F16030942224F62B1BC3D5EDA63AF0C07C12E9AF60F4EF5A29A84E9706CB3E23 | sha512: 12BDBC7F6631B02350FF0C519BA2BF14CC75C624485C67BCFA1AEF3D2DDFF4ACC6D17EDFDEBBBBD20BA95F14342F7874D8345567FC2478554FAAB03534C8EC68
tools\_internal\api-ms-win-crt-environment-l1-1-0.dll
md5: CBE0B9FBB04ADAFB76CC0D50BA1B2D1A | sha1: 40C0D5674248949A0128949C12443A72957D2CA8 | sha256: 34E9A4FFE038E13BCADBEC9A783896B3A67988B42D6353CE70D8987A0BF5E888 | sha512: BE1186E5C2AAA3A9E23E8D3F14FDF332FBBB292C81FCD75208D990FFB5D173D9ADBA41F49814DE1AB275388B228C9AC5D1EB72FD5AFCD35B39B1DD3CCA959464
tools\_internal\api-ms-win-crt-filesystem-l1-1-0.dll
md5: 15DA941B5AD1BDDF76B8F09F359AA1B6 | sha1: EA95E65EDB63AA2A6FE4BB365EA3F3BC995A4DF3 | sha256: 4F5FAF054747FD8D9A202B31BB8F687FD369C47F82AD5860DCBF7A58D2CF924F | sha512: F4936BE1EA8ABF6B89B7C26DB6C48ABBE4498A48B32382EBE74D5DFA6AC34D3B8CF6FFDCABBFE3403F3ABB19ABEC4486C39F52838CDD85BB04E8CD8E049A566F
tools\_internal\api-ms-win-crt-heap-l1-1-0.dll
md5: DB4FEA844F77012D7DB0D0923AACBBAD | sha1: 8A8A99B1F392A02AAB29EF35E8207DB3C47E930C | sha256: 0F885499CC169A02E409445D5307FCE784295F165DCB6A8E178733C789D4526A | sha512: DE5851C4043F6BF00D9EF1B30BB9DEFD4AB70FFE62605B0D602F364A2DBACF86C2DFE9C43D60BCDA57F0254E358644518D67C9133DE0D49FD795E3D4912924B8
tools\_internal\api-ms-win-crt-locale-l1-1-0.dll
md5: 8AB373179221F08C7A7F44CADAC328CA | sha1: F06818BF5CDA974D3D99456B13921B22C6470B8D | sha256: 0DFE9BE5FEF238A9EBA4D4F03B5989389373CC8D82B03D2C34D17800655BFABF | sha512: 90591B6E4A43FC8790D4F9063E5FCA2BA8076082E3A1EB00DCFF37DA1EAFF62A209699EDA000B079C2EF0EBCEAF1231CEC197800CEAD99A5912DD4AAFDF2F3C9
tools\_internal\api-ms-win-crt-math-l1-1-0.dll
md5: 5387082EFCDD2E6C6BE7ED8A88E03005 | sha1: 7078D27B3F64E49FCF2B70DA4B904A10BC14C473 | sha256: 5A4D354626564C6CDC7D28475B6342CF79025E9E4DF0F4B43DD835BEEA42A548 | sha512: 82446BBDD1B6C1459E0A034C1C031B817794B94A67FC093D63D8A644C66A9B7039AB8CBDF5383198BE66388CF91FDBBB47A9521658EAF83783970B7C2803B305
tools\_internal\api-ms-win-crt-process-l1-1-0.dll
md5: 8B22CA474685E79ABA401635419F69A4 | sha1: F6BD165297FDCE1324FFC001111D3DD18FB115CE | sha256: D95C89D8F1DCBCADB3CCC28877B66BE769A4351D4E0173F4A192128EE3D7DD51 | sha512: 6FA3A5CA83A6349B6F3C5DC31426E171816AFE1D49D4DA5E06ED20D31CEA62374D40B7E66A5929F24777DC2B107DADE2D4F6AA1D5E0BA0174C51E903A98166C9
tools\_internal\api-ms-win-crt-runtime-l1-1-0.dll
md5: 820662FEF00AF503E207218ECD67765E | sha1: FC30F9477BE473EE9F9A59A4A19BC93A835A04AA | sha256: 1A460B02EBD56AC03F5117BF3CAB74B3C83845B454881745C71818AFE55286E3 | sha512: 0C9E0BC655FE102C6F1E2AF24C38B3EBBA6284F6C21E5352304408822F3B0785530A8A97FA30850483CB05B3443EFAAB0D19E637C576497852AA094576E7EA1F
tools\_internal\api-ms-win-crt-stdio-l1-1-0.dll
md5: 2428B2DC3EF3F8E811BB0F425794C33C | sha1: F5ADF8A7E069B6331CF31DCBE105F5BB11D2BA09 | sha256: 5723612D49C760AD13CADC7857B417145406D55D0011ED55E2894F4B3BC7D4E3 | sha512: 06C30ED1FF841CCA96A18FACF2FB337066642364B2A4799301E01E022ECF110BC6FE015D8C4DA9275238DCC034FBE3B034ECC9552BA229DBDA7BA2111E22AADE
tools\_internal\api-ms-win-crt-string-l1-1-0.dll
md5: 5847CF91E8DF22423AF0F194FBC1C8AC | sha1: 1F349445DCFBE959EC44FCC2E487FC63D249F988 | sha256: 6AF625152090F685F05AD2B03F0739296D13A09C56A91A8F08461C6A22309D43 | sha512: 81EA5AEFD5B64E206B1A671FE1A22D908AD55A6DB5894018BD779A3F4F73E8358070B8DDB27B7C534F965238E364AF60C6E92B73AE07E19445822A947CFAF89B
tools\_internal\api-ms-win-crt-time-l1-1-0.dll
md5: 111DF391E83DC74F0A66C11817ABAA6A | sha1: 7BDEAC1CDD5B06FAAF20D113A239C4FFE73D2610 | sha256: EDC8849CFB5BC3D946FB0C4B86A7788A2A04D113C44E5773243C0ABF7DCA543F | sha512: 5ACAA477B10BDDC4CAEA85C0D725832124765A044E8416ADA66161278E9DDF293CA8D4AC8DCDC6622A6920B9DB7E6A62D306B7FB67DCAD1CBC9BA37F32A6BD36
tools\_internal\api-ms-win-crt-utility-l1-1-0.dll
md5: A26EE6BD274CB850636742F9A5879193 | sha1: 154FAC323F50A8CF0FA730AFA1D3A58F4B06126A | sha256: BDFEABCBFE6F9E5305F2B39F14E6819E725FEAABA14A9372DDADB9408F99757F | sha512: 845E4A06D4E7E08EA13DB32AD619D9B4726E8A62F8D52D261816F917D3C78DCB76297979FD0C6B446ADDCC93B5D48D499CD4C5734E4BC325166F42A8F18F7400
tools\_internal\base_library.zip
md5: 0CB8186855E5A17427AA0F2D16E491A9 | sha1: 8E370A2A864079366D329377BEC1A9BBC54B185C | sha256: 13E24B36C20B3DA9914C67B61614B262F3FC1CA7B2EE205DED41ACC57865BFEF | sha512: 855FF87E74E4BD4719DB5B17E577E5AE6CA5EEDD539B379625B28BCCDF417F15651A3BACF06D6188C3FCAAC5814DEE753BF058F59F73C7050A0716AA7E718168
tools\_internal\dlss_updater\auto_updater.py
import os
import sys
import json
import shutil
import zipfile
import subprocess
import time
from urllib import request
from urllib.error import URLError
from packaging import version
from dlss_updater.version import __version__
from dlss_updater.logger import setup_logger

logger = setup_logger()

GITHUB_API_URL = "https://api.github.com/repos/Recol/DLSS-Updater/releases/latest"


def check_for_updates():
    """
    Check for available updates by comparing versions.
    Returns (latest_version, download_url) tuple or (None, None) if no update available.
    """
    try:
        with request.urlopen(GITHUB_API_URL) as response:
            latest_release = json.loads(response.read().decode())
        latest_version = latest_release["tag_name"].lstrip("V")

        if version.parse(latest_version) > version.parse(__version__):
            return latest_version, latest_release["assets"][0]["browser_download_url"]
        else:
            return None, None
    except URLError as e:
        logger.error(f"Error checking for updates: {e}")
        return None, None


def download_update(download_url):
    """
    Download and extract the update package.
    Returns path to new executable or None if download/extraction fails.
    """
    try:
        update_dir = os.path.join(os.path.dirname(sys.executable), "update")
        os.makedirs(update_dir, exist_ok=True)
        update_zip = os.path.join(update_dir, "update.zip")

        logger.info("Downloading update package...")
        request.urlretrieve(download_url, update_zip)

        logger.info("Extracting update package...")
        with zipfile.ZipFile(update_zip, "r") as zip_ref:
            zip_ref.extractall(update_dir)

        os.remove(update_zip)

        # Look for the new executable
        new_exe = None
        for root, dirs, files in os.walk(update_dir):
            for file in files:
                if file.lower() == "dlss_updater.exe":
                    new_exe = os.path.join(root, file)
                    break
            if new_exe:
                break

        if new_exe:
            logger.info(f"Found new executable: {new_exe}")
            return new_exe
        else:
            logger.error("Error: New executable not found in the update package.")
            return None
    except Exception as e:
        logger.error(f"Error downloading update: {e}")
        if os.path.exists(update_dir):
            shutil.rmtree(update_dir)
        return None


def update_script(current_exe, new_exe):
    """
    Perform the actual update by replacing the old executable with the new one.
    """
    # Wait for the original process to exit
    time.sleep(2)

    try:
        # Replace the old executable with the new one
        os.remove(current_exe)
        shutil.move(new_exe, current_exe)

        # Clean up the update directory
        update_dir = os.path.dirname(new_exe)
        shutil.rmtree(update_dir)

        # Start the updated executable
        subprocess.Popen([current_exe], creationflags=subprocess.CREATE_NEW_CONSOLE)

        # Log the update completion
        with open(
            os.path.join(os.path.dirname(current_exe), "update_log.txt"), "w"
        ) as f:
            f.write(f"Update completed at {time.ctime()}. New executable started.")

    except Exception as e:
        # Log the error
        with open(
            os.path.join(os.path.dirname(current_exe), "update_error_log.txt"), "w"
        ) as f:
            f.write(f"Error during update process at {time.ctime()}: {str(e)}")


def perform_update(new_exe_path):
    """
    Start the update process in a new process and exit the current one.
    """
    current_exe = sys.executable

    # Start the update process in a separate process
    subprocess.Popen(
        [sys.executable, __file__, "update", current_exe, new_exe_path],
        creationflags=subprocess.CREATE_NO_WINDOW,
    )

    # Exit the current process
    sys.exit(0)


def auto_update():
    """
    Main update function that orchestrates the update process.
    Returns True if an update was downloaded and ready to install,
    False otherwise.
    """
    logger.info("Checking for updates...")
    latest_version, download_url = check_for_updates()

    if latest_version:
        logger.info(f"New version available: {latest_version}")
        new_exe_path = download_update(download_url)

        if new_exe_path:
            logger.info("Update downloaded successfully.")
            logger.info(
                "The application will now close and update. It will restart automatically."
            )
            perform_update(new_exe_path)
            return True
    else:
        logger.info("No updates available. You have the latest version.")

    return False


if __name__ == "__main__":
    if len(sys.argv) > 1 and sys.argv[1] == "update":
        update_script(sys.argv[2], sys.argv[3])
tools\_internal\dlss_updater\config.py
import os
import sys
import configparser
from enum import StrEnum
import appdirs
from .logger import setup_logger

logger = setup_logger()


def get_config_path():
    """Get the path for storing configuration files"""
    app_name = "DLSS-Updater"
    app_author = "Recol"
    config_dir = appdirs.user_config_dir(app_name, app_author)
    os.makedirs(config_dir, exist_ok=True)
    return os.path.join(config_dir, "config.ini")


def resource_path(relative_path):
    """Get absolute path to resource, works for dev and for PyInstaller"""
    try:
        base_path = sys._MEIPASS
    except Exception:
        base_path = os.path.abspath(".")
    return os.path.join(base_path, relative_path)


LATEST_DLL_VERSIONS = {
    "nvngx_dlss.dll": "3.17.20.0",
    "nvngx_dlssg.dll": "3.8.1.0",
    "nvngx_dlssd.dll": "3.17.20.0",
}

LATEST_DLL_PATHS = {
    "nvngx_dlss.dll": resource_path(os.path.join("latest_dll", "nvngx_dlss.dll")),
    "nvngx_dlssg.dll": resource_path(os.path.join("latest_dll", "nvngx_dlssg.dll")),
    "nvngx_dlssd.dll": resource_path(os.path.join("latest_dll", "nvngx_dlssd.dll")),
}


class LauncherPathName(StrEnum):
    STEAM = "SteamPath"
    EA = "EAPath"
    EPIC = "EpicPath"
    GOG = "GOGPath"
    UBISOFT = "UbisoftPath"
    BATTLENET = "BattleDotNetPath"
    XBOX = "XboxPath"


class ConfigManager(configparser.ConfigParser):
    _instance = None

    def __new__(cls, *args, **kwargs):
        if cls._instance is None:
            cls._instance = super(ConfigManager, cls).__new__(cls)
        return cls._instance

    def __init__(self):
        if not hasattr(self, "initialized"):
            super().__init__()
            self.logger = setup_logger()
            self.config_path = get_config_path()
            self.read(self.config_path)
            
            # Initialize sections
            sections = {
                "LauncherPaths": {
                    LauncherPathName.STEAM: "",
                    LauncherPathName.EA: "",
                    LauncherPathName.EPIC: "",
                    LauncherPathName.GOG: "",
                    LauncherPathName.UBISOFT: "",
                    LauncherPathName.BATTLENET: "",
                    LauncherPathName.XBOX: "",
                },
                "Settings": {
                    "CheckForUpdatesOnStart": "True",
                    "AutoBackup": "True",
                    "MinimizeToTray": "False",
                },
                "Updates": {
                    "LastUpdateCheck": "",
                    "CurrentDLSSVersion": LATEST_DLL_VERSIONS["nvngx_dlss.dll"],
                }
            }
            
            for section, values in sections.items():
                if not self.has_section(section):
                    self.add_section(section)
                for key, value in values.items():
                    if key not in self[section]:
                        self[section][key] = value
            
            self.save()
            self.initialized = True

    def update_launcher_path(self, path_to_update: LauncherPathName, new_launcher_path: str):
        """Update launcher path in config"""
        self.logger.debug(f"Attempting to update path for {path_to_update}.")
        self["LauncherPaths"][path_to_update] = new_launcher_path
        self.save()
        self.logger.debug(f"Updated path for {path_to_update}.")

    def check_path_value(self, path_to_check: LauncherPathName) -> str:
        """Get launcher path from config"""
        return self["LauncherPaths"].get(path_to_check, "")

    def reset_launcher_path(self, path_to_reset: LauncherPathName):
        """Reset launcher path to default empty value"""
        self.logger.debug(f"Resetting path for {path_to_reset}")
        self["LauncherPaths"][path_to_reset] = ""
        self.save()

    def get_setting(self, setting_name: str, default_value: str = "") -> str:
        """Get setting value with fallback"""
        return self["Settings"].get(setting_name, default_value)

    def update_setting(self, setting_name: str, value: str):
        """Update setting value"""
        self["Settings"][setting_name] = value
        self.save()

    def update_last_check_time(self, timestamp: str):
        """Update last update check timestamp"""
        self["Updates"]["LastUpdateCheck"] = timestamp
        self.save()

    def update_current_dlss_version(self, version: str):
        """Update current DLSS version"""
        self["Updates"]["CurrentDLSSVersion"] = version
        self.save()

    def save(self):
        """Save configuration to disk"""
        with open(self.config_path, "w") as configfile:
            self.write(configfile)


config_manager = ConfigManager()
tools\_internal\dlss_updater\constants.py
DLL_TYPE_MAP = {
    "nvngx_dlss.dll": "DLSS DLL",
    "nvngx_dlssg.dll": "DLSS Frame Generation DLL",
    "nvngx_dlssd.dll": "DLSS Ray Reconstruction DLL",
}
tools\_internal\dlss_updater\dlss_updater.log
 
tools\_internal\dlss_updater\icons\battlenet.png
 
tools\_internal\dlss_updater\icons\dlss_updater.ico
 
tools\_internal\dlss_updater\icons\dlss_updater.png
 
tools\_internal\dlss_updater\icons\dlss_updater_full.png
 
tools\_internal\dlss_updater\icons\ea.png
 
tools\_internal\dlss_updater\icons\epic.png
 
tools\_internal\dlss_updater\icons\gog.png
 
tools\_internal\dlss_updater\icons\reset.png
 
tools\_internal\dlss_updater\icons\steam.png
 
tools\_internal\dlss_updater\icons\ubisoft.png
 
tools\_internal\dlss_updater\icons\update.png
 
tools\_internal\dlss_updater\icons\xbox.png
 
tools\_internal\dlss_updater\lib\threading_lib.py
from PyQt6.QtCore import QThreadPool, QRunnable, pyqtSignal, QObject

class WorkerSignals(QObject):
    """Signals available from a running worker thread."""
    finished = pyqtSignal()
    result = pyqtSignal(object)
    error = pyqtSignal(tuple)
    progress = pyqtSignal(int)


class Worker(QRunnable):
    """
    Worker thread for running background tasks.
    
    Inherits from QRunnable for better thread pool management.
    """
    def __init__(self, fn, *args, **kwargs):
        super().__init__()
        
        # Store constructor arguments (re-used for processing)
        self.fn = fn
        self.args = args
        self.kwargs = kwargs
        
        # Create signals for communication
        self.signals = WorkerSignals()
        
        # Add an option to stop the worker if needed
        self.is_running = True

    def run(self):
        """
        Initialise the runner function with passed args, kwargs.
        
        Automatic handling of different function signatures and return values.
        """
        try:
            # Retrieve args/kwargs here; and fire processing using them
            result = self.fn(*self.args, **self.kwargs)
        except Exception as e:
            # If function raised an exception, capture it
            import traceback
            
            # Package the exception details
            exctype = type(e)
            value = str(e)
            tb = traceback.format_exc()
            
            # If still running, emit the error signal
            if self.is_running:
                self.signals.error.emit((exctype, value, tb))
        else:
            # If function successfully completed, emit result
            if self.is_running:
                if result is not None:
                    self.signals.result.emit(result)
        finally:
            # Always emit finished signal
            if self.is_running:
                self.signals.finished.emit()

    def stop(self):
        """Set the running state to False to prevent further signal emissions."""
        self.is_running = False


class ThreadManager:
    """
    Manages thread pool and worker creation for background tasks.
    
    Provides a simplified interface for running functions in a thread pool.
    """
    def __init__(self, parent=None):
        # Create a thread pool
        self.thread_pool = QThreadPool()
        
        # Set up maximum thread count (adjust as needed)
        self.thread_pool.setMaxThreadCount(1)
        
        # Store the current worker
        self.current_worker = None
        
        # Store the parent (optional)
        self.parent = parent
        
        # Signals to be connected from the current worker
        self.signals = None

    def assign_function(self, func, *args, **kwargs):
        """
        Assign a function to be run in a background thread.
        
        Stops any existing worker before creating a new one.
        """
        # Stop any existing worker
        if self.current_worker:
            self.current_worker.stop()
        
        # Create a new worker
        worker = Worker(func, *args, **kwargs)
        
        # Store the current worker
        self.current_worker = worker
        
        # Update signals reference
        self.signals = worker.signals

    def run(self):
        """
        Run the currently assigned worker in the thread pool.
        
        Adds the worker to the thread pool for execution.
        """
        if self.current_worker:
            # Add worker to thread pool
            self.thread_pool.start(self.current_worker)

    def waitForDone(self):
        """
        Wait for all threads in the pool to complete.
        
        Useful for clean shutdown of the application.
        """
        # Stop current worker if exists
        if self.current_worker:
            self.current_worker.stop()
        
        # Wait for thread pool to finish
        self.thread_pool.waitForDone()
tools\_internal\dlss_updater\logger.py
import logging
import sys
from pathlib import Path
from PyQt6.QtCore import QObject, pyqtSignal, Qt
from PyQt6.QtWidgets import QTextBrowser


class QLoggerLevelSignal(QObject):
    """Signals for the Logger QTextBrowser derived class."""
    debug = pyqtSignal()
    info = pyqtSignal()
    warning = pyqtSignal()
    error = pyqtSignal()


class LoggerWindow(QTextBrowser):
    """A QTextBrowser subclass that have signals and a dict for ease of access to said signals."""
    def __init__(self, parent=None):
        super().__init__(parent)
        self.signals = QLoggerLevelSignal()
        self.signals_to_emit = {
            "DEBUG": self.signals.debug,
            "INFO": self.signals.info,
            "WARNING": self.signals.warning,
            "ERROR": self.signals.error,
        }
        # Set document max size to prevent memory issues with very large logs
        self.document().setMaximumBlockCount(5000)  # Limit to last 5000 lines
        # Set text browser properties
        self.setReadOnly(True)
        self.setOpenExternalLinks(False)
        # Enable smooth scrolling
        self.verticalScrollBar().setSingleStep(2)


class QLogger(logging.Handler, QObject):
    """Logger handler for the Qt GUI"""
    # Define the signal as a class attribute
    logMessage = pyqtSignal(str, str)

    def __init__(self, text_browser):
        logging.Handler.__init__(self)
        QObject.__init__(self)  # Initialize QObject
        
        self.text_browser = text_browser
        self.colors_dict = {
            "DEBUG": "white",
            "INFO": "green", 
            "WARNING": "yellow",
            "ERROR": "red",
        }
        # Connect the signal to the slot method
        self.logMessage.connect(self.write_log, Qt.ConnectionType.QueuedConnection)

    def emit(self, record):
        """
        Logs the record to the text browser object.
        @param record: LogRecord object to log.
        """
        msg = self.format(record)
        # Emit the signal with levelname and formatted message
        self.logMessage.emit(record.levelname, msg)

    def write_log(self, levelname, msg):
        """Write the log message to the text browser in the main thread."""
        color = self.colors_dict[levelname]
        formatted_msg = f'<font color="{color}">{msg}</font>'
        self.text_browser.signals_to_emit[levelname].emit()
        self.text_browser.append(formatted_msg)
        # Scroll to bottom
        scrollbar = self.text_browser.verticalScrollBar()
        scrollbar.setValue(scrollbar.maximum())


def setup_logger(log_file_name="dlss_updater.log"):
    """
    Setups the initial logger.
    param: log_file_name: filename to be used for the logfile.
    return: logger instance created.
    """
    logger = logging.getLogger("DLSSUpdater")

    # Check if the logger has already been configured
    if not logger.handlers:
        logger.setLevel(logging.DEBUG)

        log_file_path = (
            Path(sys.executable).parent / log_file_name
            if getattr(sys, "frozen", False)
            else Path(__file__).parent / log_file_name
        )

        # Create handlers
        console_handler = logging.StreamHandler(sys.stdout)
        file_handler = logging.FileHandler(log_file_path, encoding="utf-8")

        # Create formatter and add it to handlers
        log_format = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s")
        console_handler.setFormatter(log_format)
        file_handler.setFormatter(log_format)

        # Add handlers to the logger
        logger.addHandler(console_handler)
        logger.addHandler(file_handler)

        # Prevent propagation to avoid duplicate logs
        logger.propagate = False

    return logger


def add_qt_handler(logger_to_extend, text_browser):
    """
    Add a QTextBrowser handler to an existing logger instance.
    @param: logger_to_extend: logger instance to be extended.
    @param: text_browser: QTextBrowser instance to be added as a logger.
    """
    # Remove any existing QLogger handlers
    for handler in logger_to_extend.handlers[:]:
        if isinstance(handler, QLogger):
            logger_to_extend.removeHandler(handler)

    # Create a new QLogger handler
    text_browser_handler = QLogger(text_browser)
    formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s")
    text_browser_handler.setFormatter(formatter)
    logger_to_extend.addHandler(text_browser_handler)


# Usage example
if __name__ == "__main__":
    logger = setup_logger()
    logger.info("This is a test log message")
    logger.info("This is a test logger.info message with an argument: %s", "test arg")
tools\_internal\dlss_updater\main_ui\main_window.py
from .. import __version__, resource_path
from ..utils import update_dlss_versions
import os
from PyQt6.QtCore import Qt, QUrl, QSize
from PyQt6.QtGui import QDesktopServices, QIcon, QPixmap
from dlss_updater.lib.threading_lib import ThreadManager
from pathlib import Path
from dlss_updater.config import config_manager, LauncherPathName
from dlss_updater.logger import add_qt_handler, LoggerWindow, setup_logger
from PyQt6.QtWidgets import (
    QMainWindow,
    QWidget,
    QVBoxLayout,
    QSplitter,
    QPushButton,
    QFileDialog,
    QHBoxLayout,
    QLabel,
    QMenu,
    QDialog,
    QTextBrowser,
)


class MainWindow(QMainWindow):
    def __init__(self, logger=None):
        super().__init__()
        self.thread_manager = ThreadManager(self)
        self.button_enum_dict = {}
        self.setWindowTitle("DLSS-Updater")
        self.setGeometry(100, 100, 600, 350)
        self.logger_expanded = False
        self.original_width = None

        # Load and set the window icon
        logo_path = resource_path(os.path.join('icons', 'dlss_updater.png'))
        logo_icon = QIcon(logo_path)
        self.setWindowIcon(logo_icon)

        # Main container
        main_container = QWidget()
        main_layout = QVBoxLayout()
        header_layout = QHBoxLayout()

        main_layout.setAlignment(Qt.AlignmentFlag.AlignTop)
        # Header section with welcome, logo, version, and other buttons
        header_left = QHBoxLayout()
        welcome_label = QLabel("Welcome to the GUI :) -Deco")
        version_label = QLabel(f"v{__version__}")
        welcome_label.setStyleSheet("color: white; font-size: 16px;")
        version_label.setStyleSheet(
            "color: #888888; font-size: 12px; margin-left: 8px;"
        )
        header_left.addWidget(welcome_label)
        header_left.addStretch()
        header_left.addWidget(version_label)

        # Add the header layout to the main layout
        header_layout.addLayout(header_left)
        main_layout.addLayout(header_layout)

        # Add the DLSS Updater logo as a separate element
        self.logo_label = QLabel()
        logo_pixmap = QPixmap(resource_path(os.path.join('dlss_updater', 'icons', 'dlss_updater_full.png')))
        logo_pixmap = logo_pixmap.scaledToWidth(int(self.width() * 0.6), Qt.TransformationMode.SmoothTransformation)
        self.logo_label.setPixmap(logo_pixmap)
        self.logo_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
        main_layout.addWidget(self.logo_label)

        # Custom folders info
        info_label = QLabel(
            "Note: For custom game folders, use any launcher button (e.g. Battle.net) and select your folder location."
        )
        info_label.setWordWrap(True)
        info_label.setStyleSheet(
            "color: white; background-color: #3C3C3C; padding: 10px; border-radius: 4px;"
        )
        main_layout.addWidget(info_label)

        # Donate, report a bug, contact, release notes, and view logs buttons
        button_layout = QHBoxLayout()
        donate_button = QPushButton("☕ Support Development")
        donate_button.clicked.connect(
            lambda: QDesktopServices.openUrl(QUrl("https://buymeacoffee.com/decouk"))
        )
        report_bug_button = QPushButton("🐛 Report a Bug")
        report_bug_button.clicked.connect(
            lambda: QDesktopServices.openUrl(
                QUrl("https://github.com/Recol/DLSS-Updater/issues")
            )
        )
        contact_button = QPushButton("📞 Contact")
        contact_menu = QMenu()
        twitter_action = contact_menu.addAction("Twitter")
        discord_action = contact_menu.addAction("Discord")
        twitter_action.triggered.connect(
            lambda: QDesktopServices.openUrl(QUrl("https://x.com/iDeco_UK"))
        )
        discord_action.triggered.connect(
            lambda: QDesktopServices.openUrl(
                QUrl("https://discord.com/users/162568099839606784")
            )
        )
        contact_button.setMenu(contact_menu)
        release_notes_button = QPushButton("📝 Release Notes")
        release_notes_button.clicked.connect(self.show_release_notes)
        view_logs_button = QPushButton("📋 View Logs")
        view_logs_button.clicked.connect(self.toggle_logger_window)
        button_layout.addWidget(donate_button)
        button_layout.addWidget(report_bug_button)
        button_layout.addWidget(contact_button)
        button_layout.addWidget(release_notes_button)
        button_layout.addWidget(view_logs_button)
        main_layout.addLayout(button_layout)

        # Original logger splitter setup
        self.logger_splitter = QSplitter(Qt.Orientation.Horizontal)
        self.can_continue = False
        self.button_list = []
        self.path_list = []

        # Launcher buttons setup
        self.setup_launcher_buttons()

        # Create QTextBrowser widget
        self.logger_window = LoggerWindow(self)

        # Set up splitter layout
        self.logger_splitter.addWidget(self.browse_buttons_container_widget)
        self.logger_splitter.addWidget(self.logger_window)
        # We want the logger_window to be collapsed by default
        self.logger_splitter.setSizes([1, 0])

        main_layout.addWidget(self.logger_splitter)
        main_container.setLayout(main_layout)
        self.setCentralWidget(main_container)

        # Set up logging
        self.logger = logger or setup_logger()
        add_qt_handler(self.logger, self.logger_window)
        self.logger_window.signals.error.connect(self.expand_logger_window)

        # Connect the update button to the threaded update function
        self.start_update_button.clicked.connect(self.call_threaded_update)

        self.apply_dark_theme()

    def show_release_notes(self):
        """Display release notes in a dialog"""
        release_notes_file = Path(resource_path("release_notes.txt"))
        if release_notes_file.exists():
            with open(release_notes_file, "r") as file:
                notes = file.read()
                dialog = QDialog(self)
                dialog.setWindowTitle("Release Notes")
                layout = QVBoxLayout()
                text_browser = QTextBrowser()
                text_browser.setPlainText(notes)
                text_browser.setStyleSheet("background-color: #3C3C3C; color: white;")
                layout.addWidget(text_browser)
                dialog.setLayout(layout)
                dialog.resize(500, 400)
                dialog.exec()

    def reset_path(self):
        """Reset the associated launcher path"""
        reset_button = self.sender()
        launcher_button = reset_button.property("reset_button")
        if launcher_button:
            launcher_enum = self.button_enum_dict.get(launcher_button.objectName())
            if launcher_enum:
                config_manager.reset_launcher_path(launcher_enum)
                launcher_button.setText(launcher_button.objectName())
                self.logger.info(f"Reset path for {launcher_button.objectName()}")

    def expand_logger_window(self):
        """Increase app window size and expands the logger window. Used only for errors."""
        if self.logger_expanded:
            return
        self.original_width = self.width()
        self.setFixedWidth(int(self.width() * 1.4))
        self.logger_splitter.setSizes([int(self.width()), int(self.width())])
        self.logger_expanded = True

    def toggle_logger_window(self):
        """Increase app window size and expands the logger window."""
        try:
            if self.logger_expanded:
                self.logger_splitter.setSizes([1, 0])
                self.setFixedWidth(self.original_width)
                self.logger_expanded = False
                return

            # Store original width before expanding
            self.original_width = self.width()

            # Expand window
            self.setFixedWidth(int(self.width() * 1.4))
            self.logger_splitter.setSizes([int(self.width()), int(self.width())])
            self.logger_expanded = True

        except Exception as e:
            self.logger.error(f"Error toggling logger window: {e}")

    def create_styled_button(
        self, text: str, icon_path: str, tooltip: str = ""
    ) -> QPushButton:
        """
        Creates styled buttons with the specific icon and tooltip.
        @param text: Text to be displayed.
        @param icon_path: Path to the icon.
        @param tooltip: Tooltip on hover. Optional.
        @return: QPushButton Created button.
        """
        button = QPushButton(f"  {text}", self)

        # Load and process icon
        icon = QIcon(resource_path(os.path.join("icons", icon_path)))
        button.setIcon(icon)
        button.setIconSize(QSize(24, 24))  # Consistent icon size

        # Set fixed height for uniformity
        button.setMinimumHeight(40)

        if tooltip:
            button.setToolTip(tooltip)

        # Connect to browse functionality if not the update button
        if "Update" not in text:
            button.clicked.connect(self.browse_folder)

        return button

    def setup_launcher_buttons(self):
        """Setups the launcher buttons."""
        # Create launcher buttons
        self.steam_text_browser = self.create_styled_button(
            "Steam Games", "steam.png", "Select Steam game locations"
        )
        self.ea_text_browser = self.create_styled_button(
            "EA Games", "ea.png", "Select EA game locations"
        )
        self.ubisoft_text_browser = self.create_styled_button(
            "Ubisoft Games", "ubisoft.png", "Select Ubisoft game locations"
        )
        self.epic_text_browser = self.create_styled_button(
            "Epic Games", "epic.png", "Select Epic game locations"
        )
        self.gog_text_browser = self.create_styled_button(
            "GOG Games", "gog.png", "Select GOG game locations"
        )
        self.battlenet_text_browser = self.create_styled_button(
            "Battle.net Games", "battlenet.png", "Select Battle.net game locations"
        )
        self.xbox_text_browser = self.create_styled_button(
            "Xbox Games", "xbox.png", "Select Xbox game locations"
        )

        # Update button with special styling
        self.start_update_button = self.create_styled_button(
            "Start Update", "update.png", "Start DLSS update process"
        )

        # Set object names for identification
        self.steam_text_browser.setObjectName("Steam")
        self.ea_text_browser.setObjectName("EA")
        self.ubisoft_text_browser.setObjectName("UBISOFT")
        self.epic_text_browser.setObjectName("EPIC")
        self.gog_text_browser.setObjectName("GOG")
        self.battlenet_text_browser.setObjectName("BATTLENET")
        self.xbox_text_browser.setObjectName("XBOX")

        # Store buttons in list
        self.button_list = [
            self.steam_text_browser,
            self.ea_text_browser,
            self.ubisoft_text_browser,
            self.epic_text_browser,
            self.gog_text_browser,
            self.battlenet_text_browser,
            self.xbox_text_browser,
        ]

        # Update button dictionary
        self.button_enum_dict.update(
            {
                "Steam": LauncherPathName.STEAM,
                "EA": LauncherPathName.EA,
                "UBISOFT": LauncherPathName.UBISOFT,
                "EPIC": LauncherPathName.EPIC,
                "GOG": LauncherPathName.GOG,
                "BATTLENET": LauncherPathName.BATTLENET,
                "XBOX": LauncherPathName.XBOX,
            }
        )

        # Create layout for buttons with reset buttons
        browse_buttons_layout = QVBoxLayout()
        for button in self.button_list:
            button_row = QHBoxLayout()
            button_row.addWidget(button, stretch=1)

            # Create reset button
            reset_button = QPushButton()
            reset_button.setIcon(
                QIcon(resource_path(os.path.join("icons", "reset.png")))
            )
            reset_button.setIconSize(QSize(16, 16))
            reset_button.setFixedSize(24, 24)
            reset_button.setToolTip("Reset path")
            reset_button.setProperty("reset_button", button)
            reset_button.clicked.connect(self.reset_path)
            reset_button.setStyleSheet(
                """
                QPushButton {
                    background-color: #4D4D4D;
                    border: 1px solid #7F7F7F;
                    border-radius: 4px;
                    padding: 2px;
                    margin: 2px;
                }
                QPushButton:hover {
                    background-color: #5A5A5A;
                }
                QPushButton:pressed {
                    background-color: #444444;
                }
            """
            )

            button_row.addWidget(reset_button)
            browse_buttons_layout.addLayout(button_row)

        browse_buttons_layout.addWidget(self.start_update_button)
        self.browse_buttons_container_widget = QWidget()
        self.browse_buttons_container_widget.setLayout(browse_buttons_layout)

    def call_threaded_update(self):
        """Start the update process in a separate thread."""
        try:
            # Disable the button immediately to prevent multiple clicks
            self.start_update_button.setEnabled(False)
            self.logger.info("Starting update process in thread...")

            # Clear any previous signal connections
            if self.thread_manager.signals:
                try:
                    # Disconnect previous connections if they exist
                    self.thread_manager.signals.finished.disconnect()
                    self.thread_manager.signals.result.disconnect()
                    self.thread_manager.signals.error.disconnect()
                except TypeError:
                    # Ignore errors if signals were not connected
                    pass

            # Assign the update function
            self.thread_manager.assign_function(update_dlss_versions)

            # Connect new signals
            self.thread_manager.signals.finished.connect(self.handle_update_finished)
            self.thread_manager.signals.result.connect(self.handle_update_result)
            self.thread_manager.signals.error.connect(self.handle_update_error)

            # Run the thread
            self.thread_manager.run()

        except Exception as e:
            self.logger.error(f"Error starting update thread: {e}")
            import traceback

            self.logger.error(traceback.format_exc())
            # Ensure button is re-enabled in case of an error
            self.start_update_button.setEnabled(True)

    def handle_update_error(self, error):
        """
        Handle errors from the update thread.
        @param error: The error from the update thread.
        """
        exctype, value, tb = error
        self.logger.error(f"Error: {exctype}")
        self.logger.error(f"Value: {value}")
        self.logger.error(f"Traceback: {tb}")
        self.start_update_button.setEnabled(True)

    def handle_update_result(self, result):
        """
        Handle results from the update thread.
        @param result: The result from the update thread.
        """
        try:
            if result:
                self.logger.info("Update process completed successfully")
            else:
                self.logger.error("Update process failed")
        except Exception as e:
            self.logger.error(f"Error handling update result: {e}")
        finally:
            self.start_update_button.setEnabled(True)

    def handle_update_finished(self):
        """Handle completion of the update thread."""
        try:
            self.logger.debug("Update thread finished")
            self.start_update_button.setEnabled(True)
            # Clean up worker reference
            self._current_worker = None
        except Exception as e:
            self.logger.error(f"Error in update finished handler: {e}")

    def closeEvent(self, event):
        """Handle application close event."""
        try:
            if self.thread_manager and self.thread_manager.current_worker:
                self.thread_manager.waitForDone()
        except Exception as e:
            self.logger.error(f"Error during cleanup: {e}")
        super().closeEvent(event)

    def get_current_settings(self):
        """Get the current settings from the settings file."""
        steam_path = config_manager.check_path_value(LauncherPathName.STEAM)
        ea_path = config_manager.check_path_value(LauncherPathName.EA)
        ubisoft_path = config_manager.check_path_value(LauncherPathName.UBISOFT)
        epic_path = config_manager.check_path_value(LauncherPathName.EPIC)
        gog_path = config_manager.check_path_value(LauncherPathName.GOG)
        battlenet_path = config_manager.check_path_value(LauncherPathName.BATTLENET)
        xbox_path = config_manager.check_path_value(LauncherPathName.XBOX)

        self.path_list = [
            steam_path,
            ea_path,
            ubisoft_path,
            epic_path,
            gog_path,
            battlenet_path,
            xbox_path,
        ]

        for i, button in enumerate(self.button_list):
            if self.path_list[i]:
                button.setText(self.path_list[i])

    def browse_folder(self):
        """Open a dialog to select a directory."""
        directory = QFileDialog.getExistingDirectory(self, "Select Folder")
        if directory:
            directory = directory.replace("/", "\\")
            self.sender().setText(directory)
            config_manager.update_launcher_path(
                self.button_enum_dict.get(self.sender().objectName()), directory
            )

    def apply_dark_theme(self):
        """Apply a dark theme using stylesheets."""
        dark_stylesheet = """
            QMainWindow {
                background-color: #2E2E2E; /* Dark background */
                color: #FFFFFF; /* White text */
            }
            QPushButton {
                background-color: #4D4D4D; /* Button background */
                color: #FFFFFF; /* Button text color */
                border: 1px solid #7F7F7F; /* Button border */
                padding: 5px;
            }
            QPushButton:hover {
                background-color: #5A5A5A; /* Button hover effect */
            }
            QTextBrowser {
                background-color: #3C3C3C; /* Text browser background */
                color: #FFFFFF; /* Text color */
                border: 1px solid #7F7F7F; /* Text browser border */
            }
            QMenu {
                background-color: #3C3C3C;
                color: #FFFFFF;
                border: 1px solid #7F7F7F;
            }
            QMenu::item:selected {
                background-color: #5A5A5A;
            }
        """
        self.setStyleSheet(dark_stylesheet)

        # Apply consistent styling to all launcher buttons
        button_style = """
                                QPushButton {
                                    background-color: #4D4D4D;
                                    color: white;
                                    border: 1px solid #7F7F7F;
                                    border-radius: 4px;
                                    padding: 8px 16px;
                                    text-align: left;
                                    margin: 2px 0px;
                                }
                                QPushButton:hover {
                                    background-color: #5A5A5A;
                                    border-color: #999999;
                                }
                                QPushButton:pressed {
                                    background-color: #444444;
                                }
                                QPushButton:disabled {
                                    background-color: #3D3D3D;
                                    color: #888888;
                                }
                            """

        for button in self.button_list:
            button.setStyleSheet(button_style)

        self.start_update_button.setStyleSheet(
            """
                    QPushButton {
                        background-color: #2D5A88;
                        color: white;
                        border: 1px solid #7F7F7F;
                        border-radius: 4px;
                        padding: 8px 16px;
                        text-align: left;
                        font-weight: bold;
                        margin: 2px 0px;
                    }
                    QPushButton:hover {
                        background-color: #366BA3;
                        border-color: #999999;
                    }
                    QPushButton:pressed {
                        background-color: #244B73;
                    }
                    QPushButton:disabled {
                        background-color: #1D3D5A;
                        color: #888888;
                    }
                """
        )
tools\_internal\dlss_updater\scanner.py
import os
from pathlib import Path
from .config import LauncherPathName, config_manager
from .whitelist import is_whitelisted
from dlss_updater.logger import setup_logger
import sys

logger = setup_logger()


def get_steam_install_path():
    try:
        if config_manager.check_path_value(LauncherPathName.STEAM):
            path = config_manager.check_path_value(LauncherPathName.STEAM)
            # Remove \steamapps\common if it exists
            path = path.replace("\\steamapps\\common", "")
            logger.debug(f"Using configured Steam path: {path}")
            return path

        import winreg

        key = winreg.OpenKey(
            winreg.HKEY_LOCAL_MACHINE, r"SOFTWARE\WOW6432Node\Valve\Steam"
        )
        value, _ = winreg.QueryValueEx(key, "InstallPath")
        config_manager.update_launcher_path(LauncherPathName.STEAM, str(value))
        return value
    except (FileNotFoundError, ImportError) as e:
        logger.debug(f"Could not find Steam install path: {e}")
        return None


def get_steam_libraries(steam_path):
    logger.debug(f"Looking for Steam libraries in: {steam_path}")
    library_folders_path = Path(steam_path) / "steamapps" / "libraryfolders.vdf"
    logger.debug(f"Checking libraryfolders.vdf at: {library_folders_path}")

    if not library_folders_path.exists():
        default_path = Path(steam_path) / "steamapps" / "common"
        logger.debug(
            f"libraryfolders.vdf not found, using default path: {default_path}"
        )
        return [default_path]

    libraries = []
    with library_folders_path.open("r", encoding="utf-8") as file:
        lines = file.readlines()
        for line in lines:
            if "path" in line:
                path = line.split('"')[3]
                library_path = Path(path) / "steamapps" / "common"
                logger.debug(f"Found Steam library: {library_path}")
                libraries.append(library_path)

    return libraries


def find_dlss_dlls(library_paths, launcher_name):
    dll_names = ["nvngx_dlss.dll", "nvngx_dlssg.dll", "nvngx_dlssd.dll"]
    dll_paths = []
    logger.debug(f"Searching for DLLs in {launcher_name}")

    for library_path in library_paths:
        logger.debug(f"Scanning directory: {library_path}")
        try:
            for root, _, files in os.walk(library_path):
                for dll_name in dll_names:
                    if dll_name.lower() in [f.lower() for f in files]:
                        dll_path = os.path.join(root, dll_name)
                        logger.debug(f"Found DLL: {dll_path}")
                        if not is_whitelisted(dll_path):
                            logger.info(
                                f"Found non-whitelisted DLSS DLL in {launcher_name}: {dll_path}"
                            )
                            dll_paths.append(dll_path)
                        else:
                            logger.info(
                                f"Skipped whitelisted game in {launcher_name}: {dll_path}"
                            )
        except Exception as e:
            logger.error(f"Error scanning {library_path}: {e}")

    return dll_paths


def get_ea_games():
    ea_path = config_manager.check_path_value(LauncherPathName.EA)
    if not ea_path or ea_path == "":
        return []
    ea_games_path = Path(ea_path)
    return [ea_games_path] if ea_games_path.exists() else []


def get_ubisoft_install_path():
    try:
        if config_manager.check_path_value(LauncherPathName.UBISOFT):
            return config_manager.check_path_value(LauncherPathName.UBISOFT)
        import winreg

        key = winreg.OpenKey(
            winreg.HKEY_LOCAL_MACHINE, r"SOFTWARE\WOW6432Node\Ubisoft\Launcher"
        )
        value, _ = winreg.QueryValueEx(key, "InstallDir")
        config_manager.update_launcher_path(LauncherPathName.UBISOFT, str(value))
        return value
    except (FileNotFoundError, ImportError):
        return None


def get_ubisoft_games(ubisoft_path):
    ubisoft_games_path = Path(ubisoft_path) / "games"
    if not ubisoft_games_path.exists():
        return []
    return [ubisoft_games_path]

def get_xbox_games():
    xbox_path = config_manager.check_path_value(LauncherPathName.XBOX)
    if not xbox_path or xbox_path == "":
        return []
    xbox_games_path = Path(xbox_path)
    return [xbox_games_path] if xbox_games_path.exists() else []

def get_epic_games():
    epic_path = config_manager.check_path_value(LauncherPathName.EPIC)
    if not epic_path or epic_path == "":
        return []
    epic_games_path = Path(epic_path)
    return [epic_games_path] if epic_games_path.exists() else []


def get_gog_games():
    gog_path = config_manager.check_path_value(LauncherPathName.GOG)
    if not gog_path or gog_path == "":
        return []
    gog_games_path = Path(gog_path)
    return [gog_games_path] if gog_games_path.exists() else []


def get_battlenet_games():
    battlenet_path = config_manager.check_path_value(LauncherPathName.BATTLENET)
    if not battlenet_path or battlenet_path == "":
        return []
    battlenet_games_path = Path(battlenet_path)
    return [battlenet_games_path] if battlenet_games_path.exists() else []


def find_all_dlss_dlls():
    logger.info("Starting find_all_dlss_dlls function")
    all_dll_paths = {
        "Steam": [],
        "EA Launcher": [],
        "Ubisoft Launcher": [],
        "Epic Games Launcher": [],
        "GOG Launcher": [],
        "Battle.net Launcher": [],
        "Xbox Launcher": [],
    }

    # Steam
    steam_path = get_steam_install_path()
    if steam_path:
        steam_libraries = get_steam_libraries(steam_path)
        all_dll_paths["Steam"] = find_dlss_dlls(steam_libraries, "Steam")

    # EA
    ea_games = get_ea_games()
    if ea_games:
        all_dll_paths["EA Launcher"] = find_dlss_dlls(ea_games, "EA Launcher")

    # Ubisoft
    ubisoft_path = get_ubisoft_install_path()
    if ubisoft_path:
        ubisoft_games = get_ubisoft_games(ubisoft_path)
        all_dll_paths["Ubisoft Launcher"] = find_dlss_dlls(
            ubisoft_games, "Ubisoft Launcher"
        )

    # Epic Games
    epic_games = get_epic_games()
    if epic_games:
        all_dll_paths["Epic Games Launcher"] = find_dlss_dlls(
            epic_games, "Epic Games Launcher"
        )

    # Xbox
    xbox_games = get_xbox_games()
    if xbox_games:
        all_dll_paths["Xbox Launcher"] = find_dlss_dlls(xbox_games, "Xbox Launcher")

    # GOG
    gog_games = get_gog_games()
    if gog_games:
        all_dll_paths["GOG Launcher"] = find_dlss_dlls(gog_games, "GOG Launcher")

    # Battle.net
    battlenet_games = get_battlenet_games()
    if battlenet_games:
        all_dll_paths["Battle.net Launcher"] = find_dlss_dlls(
            battlenet_games, "Battle.net Launcher"
        )

    # Remove duplicates
    unique_dlls = set()
    for launcher in all_dll_paths:
        all_dll_paths[launcher] = [
            dll
            for dll in all_dll_paths[launcher]
            if str(dll) not in unique_dlls and not unique_dlls.add(str(dll))
        ]

    return all_dll_paths
tools\_internal\dlss_updater\updater.py
import os
import shutil
import pefile
from dlss_updater.config import LATEST_DLL_VERSIONS, LATEST_DLL_PATHS
from pathlib import Path
import stat
import time
import psutil
from packaging import version
from .logger import setup_logger
from .constants import DLL_TYPE_MAP

logger = setup_logger()


def parse_version(version_string):
    # Replace commas with dots and remove any trailing zeros
    cleaned_version = ".".join(version_string.replace(",", ".").split(".")[:3])
    return version.parse(cleaned_version)


def get_dll_version(dll_path):
    try:
        with open(dll_path, "rb") as file:
            pe = pefile.PE(data=file.read())
            for fileinfo in pe.FileInfo:
                for entry in fileinfo:
                    if hasattr(entry, "StringTable"):
                        for st in entry.StringTable:
                            for key, value in st.entries.items():
                                if key == b"FileVersion":
                                    return value.decode("utf-8").strip()
    except Exception as e:
        logger.error(f"Error reading version from {dll_path}: {e}")
    return None


def remove_read_only(file_path):
    if not os.access(file_path, os.W_OK):
        logger.info(f"Removing read-only attribute from {file_path}")
        os.chmod(file_path, stat.S_IWRITE)


def restore_permissions(file_path, original_permissions):
    os.chmod(file_path, original_permissions)


def is_file_in_use(file_path, timeout=5):
    start_time = time.time()
    while time.time() - start_time < timeout:
        try:
            with open(file_path, "rb"):
                return False
        except PermissionError:
            for proc in psutil.process_iter(["pid", "name", "open_files"]):
                try:
                    for file in proc.open_files():
                        if file.path == file_path:
                            logger.error(
                                f"File {file_path} is in use by process {proc.name()} (PID: {proc.pid})"
                            )
                            return True
                except (psutil.NoSuchProcess, psutil.AccessDenied):
                    pass
        time.sleep(0.1)
    logger.info(f"Timeout reached while checking if file {file_path} is in use")
    return True  # Assume file is NOT in use if we can't determine otherwise to prevent hanging conditions


def normalize_path(path):
    return os.path.normpath(path)


def create_backup(dll_path):
    backup_path = dll_path.with_suffix(".dlsss")
    try:
        logger.info(f"Attempting to create backup at: {backup_path}")
        if backup_path.exists():
            logger.info("Previous backup exists, removing...")
            try:
                os.chmod(backup_path, stat.S_IWRITE)
                os.remove(backup_path)
                logger.info("Successfully removed old backup")
            except Exception as e:
                logger.error(f"Failed to remove old backup: {e}")
                return None

        dir_path = os.path.dirname(backup_path)
        os.chmod(dir_path, stat.S_IWRITE | stat.S_IREAD | stat.S_IEXEC)

        shutil.copy2(dll_path, backup_path)

        if backup_path.exists():
            os.chmod(backup_path, stat.S_IWRITE | stat.S_IREAD)
            logger.info(f"Successfully created backup at: {backup_path}")
            return backup_path
        else:
            logger.error("Backup file not created")
            return None
    except Exception as e:
        logger.error(f"Failed to create backup for {dll_path}: {e}")
        logger.error(f"Error type: {type(e)}")
        return None


def update_dll(dll_path, latest_dll_path):
    dll_path = Path(normalize_path(dll_path)).resolve()
    latest_dll_path = Path(normalize_path(latest_dll_path)).resolve()
    logger.info(f"Checking DLL at {dll_path}...")

    dll_type = DLL_TYPE_MAP.get(dll_path.name.lower(), "Unknown DLL type")
    original_permissions = os.stat(dll_path).st_mode

    try:
        existing_version = get_dll_version(dll_path)
        latest_version = get_dll_version(latest_dll_path)

        if existing_version and latest_version:
            existing_parsed = parse_version(existing_version)
            latest_parsed = parse_version(latest_version)

            logger.info(
                f"Existing version: {existing_version}, Latest version: {latest_version}"
            )

            if existing_parsed < parse_version("2.0.0"):
                logger.info(
                    f"Skipping update for {dll_path}: Version {existing_version} is less than 2.0.0 and cannot be updated."
                )
                return False, None, dll_type

            if existing_parsed >= latest_parsed:
                logger.info(
                    f"{dll_path} is already up-to-date (version {existing_version})."
                )
                return False, None, dll_type

        if not dll_path.exists():
            logger.error(f"Error: Target DLL path does not exist: {dll_path}")
            return False, None, dll_type

        if not latest_dll_path.exists():
            logger.error(f"Error: Latest DLL path does not exist: {latest_dll_path}")
            return False, None, dll_type

        if not os.access(dll_path.parent, os.W_OK):
            logger.error(
                f"Error: No write permission to the directory: {dll_path.parent}"
            )
            return False, None, dll_type

        backup_path = create_backup(dll_path)
        if not backup_path:
            return False, None, dll_type

        remove_read_only(dll_path)

        retry_count = 3
        while retry_count > 0:
            if not is_file_in_use(str(dll_path)):
                break
            logger.info(
                f"File is in use. Retrying in 2 seconds... (Attempts left: {retry_count})"
            )
            time.sleep(2)
            retry_count -= 1

        if retry_count == 0:
            logger.info(
                f"File {dll_path} is still in use after multiple attempts. Cannot update."
            )
            restore_permissions(dll_path, original_permissions)
            return False, None, dll_type

        try:
            os.remove(dll_path)
            shutil.copyfile(latest_dll_path, dll_path)
            restore_permissions(dll_path, original_permissions)

            # Verify update
            new_version = get_dll_version(dll_path)
            if new_version == latest_version:
                logger.info(
                    f"Successfully updated {dll_path} from version {existing_version} to {latest_version}."
                )
                return True, backup_path, dll_type
            else:
                logger.error(
                    f"Version verification failed - Expected: {latest_version}, Got: {new_version}"
                )
                return False, backup_path, dll_type

        except Exception as e:
            logger.error(f"File update operation failed: {e}")
            if backup_path and backup_path.exists():
                try:
                    shutil.copyfile(backup_path, dll_path)
                    logger.info("Restored backup after failed update")
                except Exception as restore_error:
                    logger.error(f"Failed to restore backup: {restore_error}")
            return False, backup_path, dll_type

    except Exception as e:
        logger.error(f"Error updating {dll_path}: {e}")
        restore_permissions(dll_path, original_permissions)
        return False, None, dll_type
tools\_internal\dlss_updater\utils.py
import os
import sys
from pathlib import Path
import ctypes
from dlss_updater.logger import setup_logger


logger = setup_logger()



try:
    from dlss_updater import (
        update_dll,
        is_whitelisted,
        __version__,
        LATEST_DLL_PATHS,
        DLL_TYPE_MAP,
        find_all_dlss_dlls,
        auto_update,
        resource_path,
    )
except ImportError as e:
    logger.error(f"Error importing dlss_updater modules: {e}")
    logger.error("Current sys.path:")
    for path in sys.path:
        logger.error(path)
    logger.error("\nCurrent directory contents:")
    for item in os.listdir():
        logger.error(item)
    logger.error("\ndlss_updater directory contents:")
    try:
        for item in os.listdir("dlss_updater"):
            logger.error(item)
    except FileNotFoundError:
        logger.error("dlss_updater directory not found")
    sys.exit(1)


def find_file_in_directory(directory, filename):
    for root, _, files in os.walk(directory):
        if filename in files:
            return os.path.join(root, filename)
    return None

def check_update_completion():
    update_log_path = os.path.join(os.path.dirname(sys.executable), "update_log.txt")
    if os.path.exists(update_log_path):
        with open(update_log_path, "r") as f:
            logger.info(f"Update completed: {f.read()}")
        os.remove(update_log_path)


def check_update_error():
    error_log_path = os.path.join(
        os.path.dirname(sys.executable), "update_error_log.txt"
    )
    if os.path.exists(error_log_path):
        with open(error_log_path, "r") as f:
            logger.error(f"Update error occurred: {f.read()}")
        os.remove(error_log_path)


def check_dependencies():
    try:
        from importlib.metadata import distributions

        required = {"pefile", "psutil"}
        installed = set()
        for dist in distributions():
            name = dist.metadata.get("Name")
            if name:
                installed.add(name.lower())
        missing = required - installed
        if missing:
            logger.info(f"Missing dependencies: {', '.join(missing)}")
            return False
        return True
    except ImportError:
        logger.error("Unable to check dependencies. Proceeding anyway.")
        return True


def run_as_admin():
    script = Path(sys.argv[0]).resolve()
    params = " ".join([str(script)] + sys.argv[1:])
    logger.info("Re-running script with admin privileges...")
    ctypes.windll.shell32.ShellExecuteW(None, "runas", sys.executable, params, None, 1)


def is_admin():
    try:
        return ctypes.windll.shell32.IsUserAnAdmin()
    except:
        return False


def extract_game_name(dll_path, launcher_name):
    parts = Path(dll_path).parts
    try:
        if "steamapps" in parts:
            return parts[parts.index("steamapps") + 2]
        elif "EA Games" in parts:
            return parts[parts.index("EA Games") + 1]
        elif "Ubisoft Game Launcher" in parts:
            return parts[parts.index("games") + 1]
        elif "Epic Games" in parts:
            return parts[parts.index("Epic Games") + 2]
        elif "GOG Galaxy" in parts:
            return parts[parts.index("Games") + 1]
        elif "Battle.net" in parts:
            return parts[parts.index("Battle.net") + 1]
        else:
            # If we can't determine the game name, use the parent directory name
            return parts[-2]
    except (ValueError, IndexError) as e:
        logger.error(
            f"Error extracting game name for {dll_path} in {launcher_name}: {e}"
        )
        return "Unknown Game"


def update_dlss_versions():
    logger.info(f"DLSS Updater version {__version__}")
    logger.info("Starting DLL search...")
    # Wrap each major operation in its own try-except block
    try:
        logger.info("Checking for updates...")
        if auto_update is None:
            logger.info("No updates were found.")
        else:
            try:
                update_available = auto_update()
                if update_available:
                    logger.info(
                        "The application will now close for the update. If the update does NOT automatically restart, please manually reboot it from the /update/ folder."
                    )
                    return  # Exit here to allow the update process to take over
            except Exception as e:
                logger.error(f"Error during update check: {e}")
                import traceback

                traceback.logger.info_exc()

        try:
            all_dll_paths = find_all_dlss_dlls()
            logger.info("DLL search completed.")
        except Exception as e:
            logger.error(f"Error finding DLLs: {e}")
            return False

        updated_games = []
        skipped_games = []
        successful_backups = []
        processed_dlls = set()

        if any(all_dll_paths.values()):
            logger.info("\nFound DLLs in the following launchers:")
            # Process each launcher
            for launcher, dll_paths in all_dll_paths.items():
                if dll_paths:
                    logger.info(f"{launcher}:")
                    for dll_path in dll_paths:
                        try:
                            dll_path = (
                                Path(dll_path)
                                if isinstance(dll_path, str)
                                else dll_path
                            )
                            if str(dll_path) not in processed_dlls:
                                # Use process_single_dll function
                                result = process_single_dll(dll_path, launcher)
                                if result:
                                    success, backup_path, dll_type = result
                                    if success:
                                        logger.info(
                                            f"Successfully processed: {dll_path}"
                                        )
                                        updated_games.append(
                                            (str(dll_path), launcher, dll_type)
                                        )
                                        if backup_path:
                                            successful_backups.append(
                                                (str(dll_path), backup_path)
                                            )
                                    else:
                                        if (
                                                backup_path
                                        ):  # If we have a backup path but success is False, it was attempted
                                            skipped_games.append(
                                                (
                                                    str(dll_path),
                                                    launcher,
                                                    "Update failed",
                                                    dll_type,
                                                )
                                            )
                                        else:  # If no backup path, it was skipped for other reasons
                                            skipped_games.append(
                                                (
                                                    str(dll_path),
                                                    launcher,
                                                    "Skipped",
                                                    dll_type,
                                                )
                                            )
                                processed_dlls.add(str(dll_path))
                        except Exception as e:
                            logger.error(f"Error processing DLL {dll_path}: {e}")
                            import traceback

                            logger.error(traceback.format_exc())
                            continue

            # Display summary after processing
            if updated_games:
                logger.info("\nGames updated successfully:")
                for dll_path, launcher, dll_type in updated_games:
                    game_name = extract_game_name(dll_path, launcher)
                    logger.info(f" - {game_name} - {launcher} ({dll_type})")
            else:
                logger.info("\nNo games were updated.")

            if successful_backups:
                logger.info("\nSuccessful backups:")
                for dll_path, backup_path in successful_backups:
                    game_name = extract_game_name(dll_path, "Unknown")
                    dll_type = DLL_TYPE_MAP.get(
                        Path(dll_path).name.lower(), "Unknown DLL type"
                    )
                    logger.info(f" - {game_name}: {backup_path} ({dll_type})")
            else:
                logger.info("\nNo backups were created.")

            if skipped_games:
                logger.info("\nGames skipped:")
                for dll_path, launcher, reason, dll_type in skipped_games:
                    game_name = extract_game_name(dll_path, launcher)
                    logger.info(
                        f" - {game_name} - {launcher} ({dll_type}) (Reason: {reason})"
                    )
        else:
            logger.info("No DLLs were found or processed.")

        return True

    except Exception as e:
        import traceback

        trace = traceback.format_exc()
        logger.error(f"Critical error in update process: {e}")
        logger.error(f"Traceback:\n{trace}")
        return False


def process_single_dll(dll_path, launcher):
    """Process a single DLL file"""
    try:
        dll_type = DLL_TYPE_MAP.get(dll_path.name.lower(), "Unknown DLL type")
        logger.info(f" - {dll_type}: {dll_path}")

        game_name = extract_game_name(str(dll_path), launcher)
        if "warframe" in game_name.lower():
            return None

        if is_whitelisted(str(dll_path)):
            return False, None, dll_type

        dll_name = dll_path.name.lower()
        if dll_name in LATEST_DLL_PATHS:
            latest_dll_path = LATEST_DLL_PATHS[dll_name]
            return update_dll(str(dll_path), latest_dll_path)

        return False, None, dll_type
    except Exception as e:
        logger.error(f"Error processing DLL {dll_path}: {e}")
        return False, None, "Error"


def display_update_summary(updated_games, skipped_games, successful_backups):
    """Display a summary of the update process"""
    logger.info("\nSummary:")
    if not (updated_games or skipped_games or successful_backups):
        logger.info("No DLLs were found or processed.")
        return

    if updated_games:
        logger.info("\nGames updated successfully:")
        for dll_path, launcher, dll_type in updated_games:
            game_name = extract_game_name(dll_path, launcher)
            logger.info(f" - {game_name} - {launcher} ({dll_type})")
    else:
        logger.info("\nNo games were updated.")

    if successful_backups:
        logger.info("\nSuccessful backups:")
        for dll_path, backup_path in successful_backups:
            game_name = extract_game_name(dll_path, "Unknown")
            dll_type = DLL_TYPE_MAP.get(Path(dll_path).name.lower(), "Unknown DLL type")
            logger.info(f" - {game_name}: {backup_path} ({dll_type})")
    else:
        logger.info("\nNo backups were created.")

    if skipped_games:
        logger.info("\nGames skipped:")
        for dll_path, launcher, reason, dll_type in skipped_games:
            game_name = extract_game_name(dll_path, launcher)
            logger.info(f" - {game_name} - {launcher} ({dll_type}) (Reason: {reason})")
tools\_internal\dlss_updater\version.py
__version__ = "2.6.3"
tools\_internal\dlss_updater\whitelist.py
import os
import csv
from io import StringIO
from urllib.request import urlopen
from urllib.error import URLError
from dlss_updater.logger import setup_logger

logger = setup_logger()

WHITELIST_URL = (
    "https://raw.githubusercontent.com/Recol/DLSS-Updater-Whitelist/main/whitelist.csv"
)


def fetch_whitelist():
    try:
        with urlopen(WHITELIST_URL) as response:
            csv_data = StringIO(response.read().decode("utf-8"))
        reader = csv.reader(csv_data)
        return set(row[0].strip() for row in reader if row and row[0].strip())
    except URLError as e:
        logger.error(f"Failed to fetch whitelist: {e}")
        return set()
    except csv.Error as e:
        logger.error(f"Failed to parse whitelist CSV: {e}")
        return set()


WHITELISTED_GAMES = fetch_whitelist()


def is_whitelisted(game_path):
    """Check if a game path matches any whitelisted games"""
    logger.debug(f"Checking game against whitelist: {game_path}")
    path_parts = game_path.lower().split(os.path.sep)
    game_name = path_parts[-1] if len(path_parts) > 1 else "Unknown"

    for game in WHITELISTED_GAMES:
        game_words = game.lower().split()
        if all(word in " ".join(path_parts) for word in game_words):
            logger.info(f"Whitelist match found: {game_name} matches {game}")
            return True

    logger.debug(f"No whitelist match found for: {game_name}")
    return False
tools\_internal\dlss_updater\__init__.py
from .scanner import get_steam_install_path, get_steam_libraries, find_dlss_dlls, find_all_dlss_dlls
from .updater import update_dll
from .whitelist import is_whitelisted
from .version import __version__
from .config import LATEST_DLL_PATHS, resource_path
from .auto_updater import auto_update
from .logger import setup_logger
from .constants import DLL_TYPE_MAP
from .lib.threading_lib import ThreadManager, WorkerSignals

__all__ = [
    "get_steam_install_path",
    "get_steam_libraries",
    "find_dlss_dlls",
    "find_all_dlss_dlls",
    "update_dll",
    "is_whitelisted",
    "__version__",
    "LATEST_DLL_PATHS",
    "resource_path",
    "auto_update",
    "setup_logger",
    "DLL_TYPE_MAP",
    "ThreadManager",
    "WorkerSignals",
]
tools\_internal\icons\battlenet.png
 
tools\_internal\icons\dlss_updater.png
 
tools\_internal\icons\dlss_updater_full.png
 
tools\_internal\icons\ea.png
 
tools\_internal\icons\epic.png
 
tools\_internal\icons\gog.png
 
tools\_internal\icons\reset.png
 
tools\_internal\icons\steam.png
 
tools\_internal\icons\ubisoft.png
 
tools\_internal\icons\update.png
 
tools\_internal\icons\xbox.png
 
tools\_internal\latest_dll\nvngx_dlss.dll
md5: AED94E7029846C356882DB166B824F5E | sha1: A9E38894E3D017722A4B6458BB9F36ED3419CCE8 | sha256: 89621E11738494356511604AE726F21C1BED1869735CC8949E9DDC8A8C2FD817 | sha512: F647943DD10B4C890D1B0E2A93FEF334F1F53E6BDDB96CFCCDBC4E9429AE4AB7017048675EB82BFE80C30BBD4F46CC5976777A1718311D746DEE81719D7A2A4B
tools\_internal\latest_dll\nvngx_dlssd.dll
md5: FDFAC845AB72D509A24EA2C16A1619C4 | sha1: 079166CF150BB7D3DAA37AF3055A82D9F9512AD2 | sha256: 9D18B5A5506910FE17BC4F032C56CB00CF1C3D622C5C8B04798E67463D6006A5 | sha512: FA70756CC253E2DEB0B63800A2E55D894FD728B04E86812E5AD26A4ED45EC004C19B8ABD9ED510D024D37E0BF6636A32A81BA34349550F3E0AF1D88094DCB4A3
tools\_internal\latest_dll\nvngx_dlssg.dll
md5: DB7B5C2C8686E1586CC56D5BDB64965C | sha1: 95990D971FF6FF2873FD68664A615666E359C294 | sha256: 8DD0AC5290E00BF846E093345AACB8A9E04CF8EAB8618CA959A12A01C02CB7B2 | sha512: 53A19985B3A783C0255E3477BCEC75EB8D1F37286BD2B1EF8E3E33B78117A647A2F5631AA6D964C89ACEACDE98B055777BC39AD91B9A15CC9766FB592EFD70D1
tools\_internal\libcrypto-3.dll
md5: 123AD0908C76CCBA4789C084F7A6B8D0 | sha1: 86DE58289C8200ED8C1FC51D5F00E38E32C1AAD5 | sha256: 4E5D5D20D6D31E72AB341C81E97B89E514326C4C861B48638243BDF0918CFA43 | sha512: 80FAE0533BA9A2F5FA7806E86F0DB8B6AAB32620DDE33B70A3596938B529F3822856DE75BDDB1B06721F8556EC139D784BC0BB9C8DA0D391DF2C20A80D33CB04
tools\_internal\libffi-8.dll
md5: 0F8E4992CA92BAAF54CC0B43AACCCE21 | sha1: C7300975DF267B1D6ADCBAC0AC93FD7B1AB49BD2 | sha256: EFF52743773EB550FCC6CE3EFC37C85724502233B6B002A35496D828BD7B280A | sha512: 6E1B223462DC124279BFCA74FD2C66FE18B368FFBCA540C84E82E0F5BCBEA0E10CC243975574FA95ACE437B9D8B03A446ED5EE0C9B1B094147CEFAF704DFE978
tools\_internal\libssl-3.dll
md5: 4FF168AAA6A1D68E7957175C8513F3A2 | sha1: 782F886709FEBC8C7CEBCEC4D92C66C4D5DBCF57 | sha256: 2E4D35B681A172D3298CAF7DC670451BE7A8BA27C26446EFC67470742497A950 | sha512: C372B759B8C7817F2CBB78ECCC5A42FA80BDD8D549965BD925A97C3EEBDCE0335FBFEC3995430064DEAD0F4DB68EBB0134EB686A0BE195630C49F84B468113E3
tools\_internal\pefile-2023.2.7.dist-info\INSTALLER
 
tools\_internal\pefile-2023.2.7.dist-info\LICENSE
 
tools\_internal\pefile-2023.2.7.dist-info\METADATA
tools\_internal\pefile-2023.2.7.dist-info\RECORD
 
tools\_internal\pefile-2023.2.7.dist-info\REQUESTED
 
tools\_internal\pefile-2023.2.7.dist-info\top_level.txt
ordlookup
pefile
peutils
tools\_internal\pefile-2023.2.7.dist-info\WHEEL
 
tools\_internal\psutil-6.1.1.dist-info\INSTALLER
 
tools\_internal\psutil-6.1.1.dist-info\LICENSE
 
tools\_internal\psutil-6.1.1.dist-info\METADATA
tools\_internal\psutil-6.1.1.dist-info\RECORD
 
tools\_internal\psutil-6.1.1.dist-info\REQUESTED
 
tools\_internal\psutil-6.1.1.dist-info\top_level.txt
psutil
tools\_internal\psutil-6.1.1.dist-info\WHEEL
 
tools\_internal\psutil\_psutil_windows.pyd
md5: 7A9632D241AD8B97BB50E8EF6DAC1CA6 | sha1: 29F0D5DE91A84FA58CF45FD134358254B7DA12ED | sha256: DD0CCDEECA681645025CA0F562EA45B5B17A1EBFCF1688CD0647A950A2992E2F | sha512: CA6AE6493961F722C07B2FACF272CAF428FD6BCD51A01C34271A18C5D898409C400E50BBAAB2771CBDC94B20041668BE8137242995C9096E511F635F1EA80BB9
tools\_internal\pyexpat.pyd
md5: AB15FFB93EA7D8ABD0A01060EEFC39F0 | sha1: F90A7FAD0C9970CD0D3E94F7B7B7959DFD6FCBC2 | sha256: FB0BE386AB5BC1695E47E49220F52BEEE99239BBE85C1749694A8A696054865F | sha512: 5DF2497C6E76C4BEE30BEF634D52A815A9112D11DBC737DFE16002D08B5BA885A56DBB56CE1DEEEAA46A99FB37D5E350A28937382DE443C553E5027BE7BEC01A
tools\_internal\PyQt6\Qt6\bin\MSVCP140.dll
md5: 01B946A2EDC5CC166DE018DBB754B69C | sha1: DBE09B7B9AB2D1A61EF63395111D2EB9B04F0A46 | sha256: 88F55D86B50B0A7E55E71AD2D8F7552146BA26E927230DAF2E26AD3A971973C5 | sha512: 65DC3F32FAF30E62DFDECB72775DF870AF4C3A32A0BF576ED1AAAE4B16AC6897B62B19E01DC2BF46F46FBE3F475C061F79CBE987EDA583FEE1817070779860E5
tools\_internal\PyQt6\Qt6\bin\MSVCP140_1.dll
md5: 0FE6D52EB94C848FE258DC0EC9FF4C11 | sha1: 95CC74C64AB80785F3893D61A73B8A958D24DA29 | sha256: 446C48C1224C289BD3080087FE15D6759416D64F4136ADDF30086ABD5415D83F | sha512: C39A134210E314627B0F2072F4FFC9B2CE060D44D3365D11D8C1FE908B3B9403EBDD6F33E67D556BD052338D0ED3D5F16B54D628E8290FD3A155F55D36019A86
tools\_internal\PyQt6\Qt6\bin\MSVCP140_2.dll
md5: 9002E0BEE6455B2322E3E717FE25F9BE | sha1: BC8DF83CC657F0F46A0BFF20565870A435ED1563 | sha256: 24B47C966B6E4A65B3E4DF866D347D3427E9BD709BE550C38224427EB5E143D3 | sha512: 28DDD087B48D5AA96EC39CCC29A4020CF75AE3C5CB6AF9A9571694D73F7AAA4FECB15336C9C7A7D12C93D8BF12EFA4FE4D8D612CD93D72C72130CAE52317D0D9
tools\_internal\PyQt6\Qt6\bin\opengl32sw.dll
md5: 83BBECF92FB68795A620B395998B131B | sha1: 026F9E87A5623FE9370C2EEDEF24C765F7312800 | sha256: B04DE4541863BC7D8879040A78889C4849C1B1DA2784C4630F734C146C2998CE | sha512: C63CA8863F63C8F415D685EFF991A1AA67E3457AC2B1F6524DB271C2986F7E79415F98F212E1F7CDD644F41BC48D558661E1136716D63F81675F664E53FDFC70
tools\_internal\PyQt6\Qt6\bin\Qt6Core.dll
md5: F6618CDB7D96415A6E11558482E32BF8 | sha1: 03F07AF673B9838C4C488C6091D878768934E654 | sha256: FCDCC96D23D1F2F2585BC17125804D4D74DE74C8BED20CE9D6CA348A14212C44 | sha512: 656F77F2A2BBB5D0E1852EFB7854A12F706854F94C6285C1D7542E760B4586F4F6558EC0968BAE2422A11D2681408853F9FEB84B9FF221F7512AB0C53724746C
tools\_internal\PyQt6\Qt6\bin\Qt6Gui.dll
md5: 244C775000161DD8C34A08C4A477C73F | sha1: CF333FFB09EE833E555E9D33AB9B31F52D2AAAEB | sha256: EF3B36AA58C41BDBEAC8EEDA14CFFFB97C2DB5165230489F2F9733A5255BF24F | sha512: 8EE6F6BDEE7D4E7F0385DD7F7B904445126839669A592B228B46F9692A94FF796072F4264D57AADFB92472E60A7A1DE0A88675633EDDD01CF2FC433EED805DDA
tools\_internal\PyQt6\Qt6\bin\Qt6Network.dll
md5: B1113A4174AAA289CCBB199A45D6AB07 | sha1: 7A2B51C4C8832426FC8C72ED854E1AA5D7CBA447 | sha256: 621FCE49FB05297962844FE20355D26322CC340C354ACB0BF1E032709C70E213 | sha512: 84C91B064A7F9E6A48D65F5436F931E61856BF58FA970204795431563F9A27BDABDF44E7875118F9A320FECA73AF9D4564B52B498779DF7B5DCDFA6D4A995BD5
tools\_internal\PyQt6\Qt6\bin\Qt6Pdf.dll
md5: 7AEFE1D8D66BC24CF968D149C3D355BA | sha1: D6BAC71C80DA7D56A042211B08A77E9E30ABF36B | sha256: 46D8E8D293F47DA45CA9F2D5D9D84C4DFABAEC2DC525B40573EE42217BE3E7CA | sha512: 329C2DEBDD3EB5CD317FEED10557001C112DF9F1F773982AFBB6C197CFBEA235A9229C11CAF0BF8CAFA1AA72A9FF80A0B3B7E1312E7ED1C98D8692B23EDB6964
tools\_internal\PyQt6\Qt6\bin\Qt6Svg.dll
md5: EB9E66A05279C8D3DEDC3F82A440D9DF | sha1: 7409A2F56313DD4999FC24F878B4242CBF0E7685 | sha256: EBF2FDD97CD098C7DFE9F4D68D0620A22F9A38F7F5FAA943EB816B07C05AF4A9 | sha512: 990798E0A48903BAD7EC8CDE195F5796D12239E3D64D9B3F9ECE4D74B71F75AB4498C1CB59248171F9CACE8970BD115FBDF1EDCF125333539C64F9313D2F2D1B
tools\_internal\PyQt6\Qt6\bin\Qt6Widgets.dll
md5: 1FCC9650B4E10AC13B280DBB8A5F435E | sha1: D9BCAC7BFA93682A8E917E450CB727754DFCC033 | sha256: 5AED2C3A8FF118747172631F9451ECB60651854AC9A90666E3B61C8454600770 | sha512: 50A389ADBAC7BD51195F5CECAFE948B2A699B6C98B0F8DBCAD220580A580869959045A6625ED7F1DB576FDF502B28BA062A3F91FF311F055359185BAA102719B
tools\_internal\PyQt6\Qt6\bin\VCRUNTIME140.dll
md5: 971DBBE854FC6AB78C095607DFAD7B5C | sha1: 1731FB947CD85F9017A95FDA1DC5E3B0F6B42CA2 | sha256: 5E197A086B6A7711BAA09AFE4EA7C68F0E777B2FF33F1DF25A21F375B7D9693A | sha512: B966AAB9C0D9459FADA3E5E96998292D6874A7078924EA2C171F0A1A50B0784C24CC408D00852BEC48D6A01E67E41D017684631176D3E90151EC692161F1814D
tools\_internal\PyQt6\Qt6\bin\VCRUNTIME140_1.dll
md5: 6BC084255A5E9EB8DF2BCD75B4CD0777 | sha1: CF071AD4E512CD934028F005CABE06384A3954B6 | sha256: 1F0F5F2CE671E0F68CF96176721DF0E5E6F527C8CA9CFA98AA875B5A3816D460 | sha512: B822538494D13BDA947655AF791FED4DAA811F20C4B63A45246C8F3BEFA3EC37FF1AA79246C89174FE35D76FFB636FA228AFA4BDA0BD6D2C41D01228B151FD89
tools\_internal\PyQt6\Qt6\plugins\generic\qtuiotouchplugin.dll
md5: BBCE418C51906DA7797BA6B0404D7F8B | sha1: F23C6BFF9EA8AB66C7355029494E22FC17A9D645 | sha256: 91E2A12E3D4C33C688B88EFCB7F85EA7986165284E440130102025A4CC050C99 | sha512: 8A17A2E8EC56268BF014A604602416441B3DA3DB50956A309F4B8DC244F9F49FFF0D642B5D7A6580A13E4406F8D934B1696877FED93B7ADBDF93FAB5C66472E5
tools\_internal\PyQt6\Qt6\plugins\iconengines\qsvgicon.dll
md5: 04E1D1FEC94098917B19A1159FF47193 | sha1: 3CD6F0171EDFB371208750E166E9E8E090FEAFE0 | sha256: FAD7B91D3DFC311E6713C9616EECF02F69D11EF6EE94E45B08623D4AA7F827AA | sha512: 6A66AAC55303898CBC2F04E413F7A1D3F8947CAC75D1B20A05A2A4ADC7133BC050797B89E5301E683CB09EC732C5FBCA8891E7539BF6B172178DD57F34F68088
tools\_internal\PyQt6\Qt6\plugins\imageformats\qgif.dll
md5: CFEB5A9AC4A431A763184B50AAF5244D | sha1: ABD689EC12D04D62B2880138C71F8976A08166EC | sha256: 466C595B87F59053DE2957581B49104C4A2CDB2E7515E061D6C45CF6C24D17AB | sha512: 935AB2B061973B931BDC13573D9DF9F63D4BC96DD13AFA566DDDAA2484B25D2058A0D037A433E94ECA45634D67BB902AA1FC395B5B8D28B6EA462F3973223D23
tools\_internal\PyQt6\Qt6\plugins\imageformats\qicns.dll
md5: 487F07BC68C42C1F84341DE31B81339E | sha1: BBA2E85D1D659B58C1748E633B00BE12230647FA | sha256: 185A84304287C558B6CF641E707257EEC49560D9AE50307E02A7DAA7AC8F8739 | sha512: 353DB60802B59AE4AF9FC6DB789AD6C7949D331AB954448B2BD1492B05C4CB9FC4A1E005BBA73A7FEE4071A6D6128CC24CC7D7281F789742DC83B89D76E645A2
tools\_internal\PyQt6\Qt6\plugins\imageformats\qico.dll
md5: 75641F83B9F6515B473E8A2C34F037BF | sha1: 1E787105B2D3F7D573EA709E07EA998020ABAFB5 | sha256: E85A0935FFA2A1BD5425DDBDC1035E8F90E8DE2E26BD0A112F86A5895BADCB86 | sha512: FCC91417BABDDF9F215D27CBDBEE5AFEE32A74FEF1DA3A5F7C763316F80BAF30DBE0E5031F24DB8913F02F27A07516FC4D3947877827AB8752F7D3661F21E13C
tools\_internal\PyQt6\Qt6\plugins\imageformats\qjpeg.dll
md5: 14139AF7CB55653167F0C021EF550419 | sha1: AF2794D2CB7E5784C37AC900988208E857E409E1 | sha256: DF30779FA1830475C7FBED9A7CA4C2A0E8F037C53BA4B08FF9B5658B1960B4C9 | sha512: 6F3E762A2F792FE89F7A052CF76E7D51EC7C528D12622B4AAD93736D4610A41E60FC1DBF3053CDB8F2B0D9B0231DE51469B55BCE483F116DD704907157EF6CF1
tools\_internal\PyQt6\Qt6\plugins\imageformats\qpdf.dll
md5: 369D2CDF81B363395F21CAEE9AB7095F | sha1: F69C78A47BAF89FEB83D82EC1BE2909290D29C5C | sha256: FFE2BB41E5AF3BD9F3800AA60ACF9A19136CCE60CB85C850C4F54BD62303C5FF | sha512: 65DA77FB310A466C7CF9EC287AA47E9ACDA8A03E68D0F0B4A1F4EBFA039643EC27A37607A4FFB783D8CDEC4E9BBFF210059B4BCB1987FCDC883A9BFF2AA7C9C6
tools\_internal\PyQt6\Qt6\plugins\imageformats\qsvg.dll
md5: 31BB65616CB6BD7C14F7800EF35E4B0B | sha1: 605451C1199D907F602D8D91D56D1E77AA39F475 | sha256: 5CE4E822A9AB127779CE33CBBD400D311192EC2E8D33F820FE0DEF27C8D03FE5 | sha512: 404E49E4422334A5D585CDEDCEAD10703A815FFAA771D32C0B1C53BBA1E38FCE01F251F0993F7ABCAF2C7591FA029152032E040C4B68FE7EA678DD1C4CD4C2EA
tools\_internal\PyQt6\Qt6\plugins\imageformats\qtga.dll
md5: 01E6729E86BE5C3E3B14C4B50D84FB78 | sha1: CA5B4430E2CCBB6E21E61AB4A8B1907F9E65A9E2 | sha256: 8F04B3B032C801FF77838B5C8CB53D774BED1A2D8262984389BA99E4FDBF30D0 | sha512: CC1A6C4A88999FF4686535B115AA8E4E63BA1DF623B41F085B97474730936C80408060C4BE482BB9358F53DF8B56336177F1FF21521F0A22786953E1F12ED308
tools\_internal\PyQt6\Qt6\plugins\imageformats\qtiff.dll
md5: 5C421747682ADAC0511A891E109882FE | sha1: 297C8516C0A1EABF0261FDD4E69DF75BEBB0FEC9 | sha256: 553FFC35CD2BAD6FC8389F23253026E7B7C1FE21FF717310ADE1011B7EED1061 | sha512: 792C3452D1CEECC14CC1E9787340E1D72CA85E1A5D5BD244B8AC713CE9AE5771AF6E98DD9ECA2A1E48E92385A658DA32C8018C609F93416E9FEBC732BD57EEF0
tools\_internal\PyQt6\Qt6\plugins\imageformats\qwbmp.dll
md5: 2D13ADB190D04B71420F85B63ED1297C | sha1: F723111A66E54B2DE44C11DDD21B9BFF23C15F09 | sha256: FA8DCB17CB14AB3DF317E3B28754DB985491703B6EB0704FA29B4414EB8252FA | sha512: 8CE429A8380E3AB5130BAA55A794C3E51C5F3B7B2BE12C0F5094777D89A107F22ACE6BC5FA4A759FF36BC08B69CC0E0E8AE61F350DCBEFC47B08BAA8B583B64B
tools\_internal\PyQt6\Qt6\plugins\imageformats\qwebp.dll
md5: 596994BCCBF0677A1656F5A7D5DCF032 | sha1: 7C4B9C48455280BD275BA8E1C5EE6B1DAD897379 | sha256: 923DCDD83563740F8C16E8AA8CFA394DC22F570FE64A407C534BDF2258902126 | sha512: 37000D69405ECDC0816C533BF33DBD1EE5D05978BEEDAD49C2840D905759863478777B728284D45A217770E7BB3984ABE694EBFF08E319B72603A1CD628450B4
tools\_internal\PyQt6\Qt6\plugins\platforms\qminimal.dll
md5: F534EA5B2E138CF01FC63095BFED91CD | sha1: 24549EB531B077DF71A3AACCD7B9E91060049BA5 | sha256: 1F03F0A4358061B49ADE4885526DC9F4C4542B130B6BC418C60328D19A3AF553 | sha512: 903B925BC5EFE8122D3CD9A78977931FF3217E3F0137E383AB898E3FA3F088E6802B0F58F324E6B43B152158B528F17EC9D399C0D78E8B25A1B0B5A6CF1BD674
tools\_internal\PyQt6\Qt6\plugins\platforms\qoffscreen.dll
md5: 712BE32D69CCFCC5E8E07EB0B0252838 | sha1: B88372F5536D44C1D9D82FCF6C5CCADA91535891 | sha256: 01EB833A4B517073B8F443F5A99198B021CAEB207773FF405B8A421D6ED405A4 | sha512: 1F940942B80B5287272D237E6CDC7E488D322822EBC96279A922E21876B493D6BC61B520ECC3FAF23F802AF070A44D09B418EDEB28CB8C7F31962A43D7E44105
tools\_internal\PyQt6\Qt6\plugins\platforms\qwindows.dll
md5: 9E9EDE62DAC5ED360AD811C409B96EC7 | sha1: C7DC498254667728A31A2C6EBB0786708CC87E1D | sha256: F4A118AFD9259CD928F55FA87F6FBC2CD88478E79BC75F4032D650B7E01E16AA | sha512: 0D6B56B68B045FFB31E5C7B81C1AAAC78078A3F4DBD53CD569C1DF350132BDD7031847573B204AB0A0B05D1695C986E8E79A5EF4617A75C5ECB9E12231606C3B
tools\_internal\PyQt6\Qt6\plugins\styles\qmodernwindowsstyle.dll
md5: A4A3C1223C9ADB54B324586C5EE374C4 | sha1: 0797C48A2946583F29F9B0A41EC770451A3CDF6C | sha256: EC2F0B95FA11C4DFD30DB27A8E593E7FEC4281EFC1BDDB578EB4DC15B8A66DF0 | sha512: D32A572B40541D90114F694B7C14BBAD67B7EB1391D56F827A83F60F78C1235D789DAD0301196A4AB404C8B1065840AAE0E5894F7EA200577441AC225AB8019B
tools\_internal\PyQt6\Qt6\translations\qtbase_ar.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_bg.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_ca.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_cs.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_da.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_de.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_en.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_es.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_fa.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_fi.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_fr.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_gd.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_he.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_hr.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_hu.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_it.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_ja.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_ka.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_ko.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_lv.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_nl.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_nn.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_pl.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_pt_BR.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_ru.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_sk.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_tr.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_uk.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_zh_CN.qm
 
tools\_internal\PyQt6\Qt6\translations\qtbase_zh_TW.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ar.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_bg.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ca.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_cs.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_da.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_de.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_en.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_es.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_fa.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_fi.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_fr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_gd.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_gl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_he.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ar.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_bg.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ca.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_cs.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_da.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_de.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_en.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_es.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_fr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_gl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_hr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_hu.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_it.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ja.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ka.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ko.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_nl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_nn.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_pl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_pt_BR.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_ru.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_sk.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_sl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_tr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_uk.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_zh_CN.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_help_zh_TW.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_hr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_hu.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_it.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ja.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ka.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ko.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_lt.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_lv.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_nl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_nn.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_pl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_pt_BR.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_pt_PT.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_ru.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_sk.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_sl.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_sv.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_tr.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_uk.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_zh_CN.qm
 
tools\_internal\PyQt6\Qt6\translations\qt_zh_TW.qm
 
tools\_internal\PyQt6\QtCore.pyd
md5: C81647CFF66445F6520F4881B4D853D9 | sha1: F25458A223DE5119C6960C0F1F493FE8D1EA0AB1 | sha256: 94440B0412810D556BAFB1B92824D8531BFBB3D593E3460D6B19483322539D9D | sha512: 6C86D4759389A25B5AEBC33149D327455E83EA6A4DE3D575E5431CB7E8167824AA25CC659AE642DED6F35083E1BBC4416327D83E3471BD509A612CE0D211CC76
tools\_internal\PyQt6\QtGui.pyd
md5: 008E1FE88263CDC69853DCDF2B297D04 | sha1: 84263829BBC2E4C030B8E5C9AF917C830E275ADF | sha256: 7A551684224EC52D2442CDF5FFA48A2300508AF8033175CAB469E0DD1975AFB9 | sha512: EC0DEDBEFD1B80B7C2C3BD7145F85353A0DF7318B2D2CF2825EF715D886E75A425E0FB9DDD5FA30D4D554DA2FE2FE2F3B5244DC264288DC86F5180AE9A0B5944
tools\_internal\PyQt6\QtWidgets.pyd
md5: D1FC9249F9633DDD6F65D914993D06B3 | sha1: A7BA3F43564D42313A5527C34A0FDF0185AA0067 | sha256: BB901B3335872D2033C4AF288C08B7E91010DC6B87F6F4E0F0264C9A9CDF6E71 | sha512: D426DC807BA6A4FB65272DF84F3FE3695DA04793B6FDBC443352FF0572B39CE7DF5B72A9A34472A4892D6B2EC078215839F63B5F04D25E9CBB4B307E8772C9C4
tools\_internal\PyQt6\sip.cp312-win_amd64.pyd
md5: 287ADA230DC00060C7A1BA26FFDF5073 | sha1: A455AF513C8B693E3CEB64B7912BFAB8F564CC95 | sha256: BA46122020B7668E16C25CCE3BE9C9273BB52033A31F5BEBD52FF5EDD772B18A | sha512: F15CFB1DA1E80632DDA2E3B7AEB2B1706107598E56B2EBF95CB941D686EEF1CD077DEEC95EE920EAFA5FD7E591B0CE787F64CE34D0F2E90CA2E191B19F09E796
tools\_internal\python3.dll
md5: ACD6BAD0EA9A91CE06189BB63D594B41 | sha1: 46EE5089000B5C312739A909662142104D4D8BE9 | sha256: 7C3E2956271EFF4949145D14635C0CA659DB5ED19215201D2D8B3A4A3D3006F9 | sha512: 79B888BDA9804A9091F5FE8D411F2A81439D3D9618E6BD73A3F729BC977CD8E15914F3FD9F90462331AEE431713C8ED7FFE864C975FAA6083D7925D17D43B315
tools\_internal\python312.dll
md5: F9A43765B486C561BF0895EB9390ED1F | sha1: B398FBD02BD7FDB32DCB88F11758A0A9826B75A4 | sha256: 3B56FA10D3797C231468CEE42CAEAAAFF40DBEDE7BC0D142EC4878493F48E07D | sha512: F2709BA81FE1E01789FC0AAE65D31F5ADCFD64DD72D161B4CDDFA35F91EB2C8D66954925C825B22CE9034FD894EE18500B1FF0A32E4D585491E09D2C540A305C
tools\_internal\pywin32_system32\pywintypes312.dll
md5: DA0E290BA30FE8CC1A44EEEFCF090820 | sha1: D38FCCD7D6F54AA73BD21F168289D7DCE1A9D192 | sha256: 2D1D60B996D1D5C56C24313D97E0FCDA41A8BD6BF0299F6EA4EB4A1E25D490B7 | sha512: BC031D61E5772C60CBAC282D05F76D81AF1AA2A29A8602C2EFA05FC0CE1079390999336237560B408E6539A77C732F5066C1590B7FEAEDB24BAA9371783F2A8F
tools\_internal\release_notes.txt
Version 2.6.0:
Addition of iconography to the GUI.
Migration to a new thread managing model to improve processing times.
Addition of support for the Xbox game launcher.
Refactoring of base code to cleanup and improve functional logic.
Remork to the whitelist logic to prevent spamming of the console with what games are whitelisted repeatedly.

Version 2.6.1:
Hotfix to fix folders being unclickable.

Version 2.6.2:
Addition of a toggle logger button to the GUI.
Coloured filtering for log events is now included.
Cleanup of the launcher icons to be more presentable.
Breakup of individual code components.
Inclusion of a "Reset path" toggle for launchers.
Inclusion of being able to view (these) release notes from inside the GUI.
Fixes to thread management shenanigans.
Bug fix to allow multiple reruns.

Version 2.6.3:
Addition of a logo.
tools\_internal\select.pyd
md5: 208CEF2D2C60A3457B13AA428DCEB347 | sha1: F78DDBE4BE3FF0B21F06AEA5F2266625D489470B | sha256: FCAC4646FF709AC07AEE532C4612A19B7070F2DD6EF67BA09C743644E92F7376 | sha512: 8B0D14D1C93089368D34BDF49602F4C8E1DAA1711C2760EEB2C59A10DBF7611FCE098AF0F11D7D5AA53D7D07DDE39B1F31ECF5F62F7F91F31D7ABEED5D828B19
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\INSTALLER
 
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\LICENSE
 
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\METADATA
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\RECORD
 
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\REQUESTED
 
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\top_level.txt
importlib_metadata
tools\_internal\setuptools\_vendor\importlib_metadata-8.0.0.dist-info\WHEEL
 
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\entry_points.txt
[console_scripts]
wheel=wheel.cli:main

[distutils.commands]
bdist_wheel=wheel.bdist_wheel:bdist_wheel

tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\INSTALLER
 
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\LICENSE.txt
MIT License

Copyright (c) 2012 Daniel Holth <[email protected]> and contributors

Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
OTHER DEALINGS IN THE SOFTWARE.
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\METADATA
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\RECORD
 
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\REQUESTED
 
tools\_internal\setuptools\_vendor\wheel-0.43.0.dist-info\WHEEL
 
tools\_internal\ucrtbase.dll
md5: CCE9B64D0F98C2370A2DA82AA9A501E0 | sha1: 0121A2B000B9A0B3F3B6660B39536FE8D72BA222 | sha256: 5D69CCE34D22D26BC6DCB4C3E58DBAE83346EB3EA203CB80769AD4C077424C96 | sha512: 66553C524CA07C537D0E7B18EA35AE0B9218D1ADF076726D4EA9071B5EC546FFD87BC6EFB55671109041A9AA007F7E0F59462341F365E448BE9071D714B6A6F9
tools\_internal\unicodedata.pyd
md5: 5E432CCA03CD6C18CF4043ED1F3AF40A | sha1: F418BC194C3D35298028BB43DC8CFF720360EB2B | sha256: E7FE7AE7342B1DEC8DFE52A95D768039A46189209B9F42A21C4D2473FAAA1753 | sha512: 02076EF322D23BECFE3E24CCA5E868A3ACB28086DFE83188D82A1A138AD24B9EA097DF667BCF8B5486B3306ABFF0138462E757171A92CD2B58A98A2AC152EC91
tools\_internal\VCRUNTIME140.dll
md5: 862F820C3251E4CA6FC0AC00E4092239 | sha1: EF96D84B253041B090C243594F90938E9A487A9A | sha256: 36585912E5EAF83BA9FEA0631534F690CCDC2D7BA91537166FE53E56C221E153 | sha512: 2F8A0F11BCCC3A8CB99637DEEDA0158240DF0885A230F38BB7F21257C659F05646C6B61E993F87E0877F6BA06B347DDD1FC45D5C44BC4E309EF75ED882B82E4E
tools\_internal\VCRUNTIME140_1.dll
md5: 68156F41AE9A04D89BB6625A5CD222D4 | sha1: 3BE29D5C53808186EBA3A024BE377EE6F267C983 | sha256: 82A2F9AE1E6146AE3CB0F4BC5A62B7227E0384209D9B1AEF86BBCC105912F7CD | sha512: F7BF8AD7CD8B450050310952C56F6A20B378A972C822CCC253EF3D7381B56FFB3CA6CE3323BEA9872674ED1C02017F78AB31E9EB9927FC6B3CBA957C247E5D57
tools\_internal\win32\win32api.pyd
md5: E9D8AB0E7867F5E0D40BD474A5CA288C | sha1: E7BDF1664099C069CEEA18C2922A8DB049B4399A | sha256: DF724F6ABD66A0549415ABAA3FDF490680E6E0CE07584E964B8BFD01E187B487 | sha512: 49B17E11D02AE99583F835B8ECF526CF1CF9CEAB5D8FAC0FBFAF45411AC43F0594F93780AE7F6CB3EBBC169A91E81DD57A37C48A8CD5E2653962FFBDCF9879BB
tools\_internal\_asyncio.pyd
md5: AD364098AA105E804C8C98D343EC3527 | sha1: 4EBE5696875C94BDBBD6CD031C0593AE89D176AF | sha256: BAC2870D61A72BE50CABC968BE924147132036185B3B538AA3B1B43D8C828A2E | sha512: BEB0A44A44969ABF7290FC560DECEAC5BD8CEE96D560041F783DD439A5F73B340634E451797AC1B67405F655299FCE5A20627CBACBA4943F7723778FDE61497F
tools\_internal\_bz2.pyd
md5: 03BAD8289D9AA18E859ED7270A719E92 | sha1: EE371B1AFC7D5BE754553D54BD0DB8968568C703 | sha256: 178C6EEB30843E656CC407AAF53AE6D0F170966E4E0BDD2EE1BEDE73962275CD | sha512: 5267906B6DAF79E898F87D24B6C0467F75DF4C53FE0F44C5DCB02EEFFEF3B9B7078B2B1AF3417BA92AB7586A7968A6B4FDB44B398287CDC1CB0C1DFF2E4E9AAB
tools\_internal\_ctypes.pyd
md5: F8D2950D5496D3940AEF6758C9E9E576 | sha1: ED68C88C14E44871A085E93BF8CF6AABC816CE28 | sha256: 9FFDEDD0F1F09F21870BD75C08D05C32994A1193BE3955E367F260690A36CBD0 | sha512: AB25FF65ABD64F39B156F7AA91C35A327C930F31D3A5D128E67E00C6307E0A0637595AB812931DC2FFEC7102E33A2AFC746DE6267F6130D4F5A8D3445BCDED79
tools\_internal\_decimal.pyd
md5: C68FC0D5C1878D02069503280234E969 | sha1: 98C8E90BF27067A4FF0764C23B95A8B845869736 | sha256: 847E2B2C69CA623E0F96BBDA0F421CA978FBB5925BEEC4CC5E4C5D9C966C4BBC | sha512: FF745D92BEB4C850371B1251C0D81C301A6CE5F6D7D5CB785A5D50DBE17817455EAFC32BB6624B24DD074270892FB3FD52A864FCCA6E6EA6D46723D3A1C2B6FA
tools\_internal\_elementtree.pyd
md5: CC59A1CA9CF077F345A6872BE765D63A | sha1: 994F6A8FA1FECCE64BEC7222D5437DF0B99F5F76 | sha256: 9DEAD134CCD6C3C2E2BDBED8A02B5853200DEB11AA2A22B926F9F2955D3F5143 | sha512: B005D2AB4072AE438551897B359F65B4A438404303EF752768B25BF754FD6A4C29FE26392CFF47DACA8255B83E095D674A768F294E4290D0F6861CFD1ADFD1EE
tools\_internal\_hashlib.pyd
md5: 5B08C2DCBE1B1DEA46ABBD6C9425878E | sha1: 8FE7D98D8F3CCAB36D92DA7E84CDC0F7D10DBADD | sha256: 823717926ADCE6B36F9C13B6555EAEAF5714C4756828F11CFFC1CE0BEF970A7E | sha512: 3843966697558800C639037A34DA13F6BD6A0C3816BC0BBD292F04DB13ACC03F0B79A8FAB4C58AD25FE18DD11471BC9DC20CC0D9C97C4C63FBE025EA6242B4D7
tools\_internal\_lzma.pyd
md5: 02B16654F73129B1A1220C00D757112D | sha1: 7CE602E4C2854BE7C6B439BC64162D58B1467750 | sha256: F6E0C786395CCC7B22F4C435B19F7073EB5974EB29E09F4C91060CAB10450E6A | sha512: 8DAF5D68230D0340E5D940067AFCA39922078CE94B6753F2BE7AE208D7982E87F5273ACA768465B110477F83735F4C8E14F7BF95C2A48E2E5457BC723398E541
tools\_internal\_multiprocessing.pyd
md5: 3966D8FD4D83FA54DB28338FF6087E08 | sha1: 61309A8044BEF8B0AA0A7447D3AFDFD7502FFEBB | sha256: B66A5A1A4AA1F187CC349E7548C3CF6D815552937A6A1D33ECC87B76794A1939 | sha512: 2203F5E547C74DC0AA177F2708AA09950825F16CFACADF84DE005296BAF305F08D732EE1E38DAFBB80EEEC72BF090B56359F5E2DF8C28C87DED8EA5552FECF71
tools\_internal\_overlapped.pyd
md5: 1E467F86CE9BB4DBE682BC0BF41B018A | sha1: 3A9D3AD208FB41A843D9D4D10F245D6A21266FFC | sha256: 0E6081DA5BBF8FCD756B02153AB7FCA12BD0BC04842ABE5765F7409AF4987B17 | sha512: 22271F1C2F709E764D88DBF6FD0EDE850F1D2D80992049B88B11F6620E6B055F18F1C55DF43D6281CCF8CA84224AC88F72CB03BEEDD9173888A71A5549B3AC59
tools\_internal\_queue.pyd
md5: 6F8624F0746FA31CF72EF568D6A121F1 | sha1: 1054FB373EA2AA51CB04FE98E6EFD130BE34FC43 | sha256: 37622CA591FB8E45A894DB9C0DA99BFCB18A820A48F028E4949D9256B69247E3 | sha512: E52199265A0360027D21034060984DD2100B79FAEB03B86CB7A6545DA10C00D860117598F659CA38C5BDDF368D2F525E4A337011CC78BBCE5307BAA52588FB9A
tools\_internal\_socket.pyd
md5: DC5A5AB89E6E2B48CB50B463B214FD89 | sha1: F7740FF9D75767CCDAC7AA0DDE8659D3133249A2 | sha256: 0E2C1089974A2757426DAC3295201A33C990C36F3C09593F8A2B6E07FD36B99C | sha512: 98A886DAA82848BC4CA352EC7E7F663822CBAD5CD40C5EA7915821097964167D0628AF47B8F19F22A825F9696F3199FFBAA94D699AE773A7EB57AE6DC1C3814F
tools\_internal\_ssl.pyd
md5: 25D4B1C6CA053C573A55D68AE3DB5CE2 | sha1: ED6BC021E3D5E5CA8083E22B718BC89D55D3AB7F | sha256: 81344E3E16CF6F2D5B24CC0CF92E95C5FD0592E4A3859BB00C3F5891E2482128 | sha512: 396A5432461408059CCDA92CC9CA641E68150AAD02EFCB635DCD3F7D68D9DD09CCEC0619705ABF3F43A959997B4F8821B726822B89C217C4054F8E8A67EB7FC3
tools\_internal\_uuid.pyd
md5: 7F64EE67CBE4066246E3B98844A781F9 | sha1: DAB90BF194C51855FFF39A8DA81DA39D47389A36 | sha256: 20557961AC93AE00829B0865018BCC565E41B52B103D2AA58405A0208ADA148F | sha512: A19546F5444C68EF3227DD3BA643D09FA005A9C3E65AFFFDB5346DC142395E8FB61163764FF255544DAE81D305D410CEC6620B97030F82F2970F33BE763AF551
tools\_internal\_wmi.pyd
md5: E203A46E89F443646CEC65D96ACEADBA | sha1: 6EC891960AC4B9D501E593157F5E89BCDDEB0CB4 | sha256: AF1077D6377D5A0AEA123F0C324CC6D151AC4A29A84AAE23A6936B6D1C64B70A | sha512: 5678C0CFF56E41B59577C13FD206AF279831BFCA6FA9E56B4FD6F960F3C11834775498ED2D9008A21B6EF65CB2DFFFA0152A4E2AD1DA4CF198ECCEEDCB599E4D

Log in or click on link to see number of positives.

In cases where actual malware is found, the packages are subject to removal. Software sometimes has false positives. Moderators do not necessarily validate the safety of the underlying software, only that a package retrieves software from the official distribution point and/or validate embedded software against official distribution point (where distribution rights allow redistribution).

Chocolatey Pro provides runtime protection from possible malware.

Add to Builder Version Downloads Last Updated Status
DLSS Updater 2.6.6 9 Thursday, January 30, 2025 Approved
DLSS Updater 2.6.5 13 Saturday, January 25, 2025 Approved
DLSS Updater 2.6.4 142 Monday, January 13, 2025 Approved
DLSS Updater 2.6.3 12 Thursday, January 9, 2025 Approved
DLSS Updater 2.6.2 44 Friday, January 3, 2025 Approved
DLSS Updater 2.6.1 27 Saturday, December 28, 2024 Approved
DLSS Updater 2.5.2 49 Thursday, December 19, 2024 Approved
DLSS Updater 2.5.1 44 Saturday, December 14, 2024 Approved
DLSS Updater 2.2.6 25 Thursday, November 14, 2024 Approved
DLSS Updater 2.2.3 155 Thursday, September 26, 2024 Approved

This package has no dependencies.

Discussion for the DLSS Updater Package

Ground Rules:

  • This discussion is only about DLSS Updater and the DLSS Updater package. If you have feedback for Chocolatey, please contact the Google Group.
  • This discussion will carry over multiple versions. If you have a comment about a particular version, please note that in your comments.
  • The maintainers of this Chocolatey Package will be notified about new comments that are posted to this Disqus thread, however, it is NOT a guarantee that you will get a response. If you do not hear back from the maintainers after posting a message below, please follow up by using the link on the left side of this page or follow this link to contact maintainers. If you still hear nothing back, please follow the package triage process.
  • Tell us what you love about the package or DLSS Updater, or tell us what needs improvement.
  • Share your experiences with the package, or extra configuration or gotchas that you've found.
  • If you use a url, the comment will be flagged for moderation until you've been whitelisted. Disqus moderated comments are approved on a weekly schedule if not sooner. It could take between 1-5 days for your comment to show up.
comments powered by Disqus