100 Commits

Author SHA1 Message Date
70b723b781 Merge pull request 'v3.2.1 Hotfix - Fixed Discord role issues on SKU change + language fallback strategy' (#14) from wip into main
Reviewed-on: #14
2025-07-10 10:08:28 +02:00
62ef15c928 Fixed Discord role issues on SKU change + language fallback strategy 2025-07-10 08:06:27 +00:00
fe3e8f9609 Fixed README and docker-compose variables 2025-07-10 07:30:21 +00:00
a95d1d41d5 Merge pull request 'wip' (#13) from wip into main
Reviewed-on: #13
2025-07-10 00:01:22 +02:00
b2789c8942 Fixed Readme 2025-07-09 22:00:06 +00:00
998c7dd500 Fixed Readme 2025-07-09 21:59:01 +00:00
4699f686a2 Localization (EU languages) 2025-07-09 21:55:00 +00:00
9bb8e8d3d7 Merge pull request 'Python 3.11 - Alpine + loop improvement' (#11) from wip into main
Reviewed-on: #11
2025-07-09 18:14:14 +02:00
66c4146223 Python 3.11 - Alpine + loop improvement 2025-07-09 16:13:12 +00:00
92edccd8d8 Merge pull request 'Multi GPU merge' (#10) from wip into main
Reviewed-on: #10
2025-07-08 19:15:29 +02:00
a2561a79b8 Fixed README 2025-07-08 16:58:26 +00:00
d4735e32b3 Fixed README 2025-07-08 16:57:49 +00:00
28991dd7a7 Fixed summary 2025-07-08 16:34:49 +00:00
781fb1270a Fixed summary 2025-07-08 16:32:29 +00:00
0eb569f7ec Updated README 2025-07-08 16:28:14 +00:00
be263e8cd1 One role per GPU support 2025-07-08 15:54:56 +00:00
6c4be66f3d Multi GPU support 2025-07-08 14:34:55 +00:00
7ca0483cfd Fixed readme 2025-07-08 09:34:06 +00:00
c8dcafbe01 v2.6 - new compose + new python execution process 2025-07-08 09:29:22 +00:00
1f9a237fce Added role env var 2025-07-08 09:15:56 +00:00
7280146d5e Removed french discord link 2025-07-07 14:12:58 +00:00
f86c56b7d1 Merge branch 'wip' 2025-07-06 13:12:35 +00:00
afbc79bee4 Github warn 2025-07-06 13:08:20 +00:00
9c2825736e Merge pull request 'wip - In english please' (#9) from wip into main
Reviewed-on: #9
2025-06-03 15:21:36 +02:00
bfd8428073 Fixed logo 2025-06-03 13:19:02 +00:00
b0eef8b8df Logo translated 2025-06-03 13:17:10 +00:00
e2d86cb787 Fully translated 2025-06-03 12:58:34 +00:00
8124ad6a2d In english please 2025-06-03 12:41:49 +00:00
27966c04e5 Merge pull request 'Correction structure, orthographe, emoji' (#8) from wip into main
Reviewed-on: #8
2025-04-24 12:53:48 +02:00
17ade38bcd Correction structure, orthographe, emoji 2025-04-24 10:51:52 +00:00
2f8ae16533 Merge pull request 'Correction variables par défaut' (#7) from wip into main
Reviewed-on: #7
2025-04-23 21:12:32 +02:00
80fe61f91d Minor changes 2025-04-23 18:58:15 +00:00
f07ff328e5 Merge pull request 'wip' (#6) from wip into main
Reviewed-on: #6
2025-04-23 19:00:54 +02:00
6657c03b07 Minor changes 2025-04-23 16:59:04 +00:00
9dae286965 Mise à jour de la documentation 2025-04-23 16:57:35 +00:00
80442f8a42 Merge pull request 'wip - correction suite modif API nvidia qui ne filtre plus' (#5) from wip into main
Reviewed-on: #5
2025-04-23 15:51:44 +02:00
43518d7d86 Correction sur le stockage des variables 2025-04-23 13:47:24 +00:00
4d6d6a259e Recherche du SKU/UPC avec le PRODUCT_NAME suite à maj de l'API de nvidia qui ne filtre plus 2025-04-23 12:46:14 +00:00
82fc77a9f4 Actualiser LICENSE 2025-03-31 23:44:50 +02:00
0822e309db Merge pull request 'Fixed footers' (#4) from wip into main
Reviewed-on: #4
2025-03-28 19:26:55 +01:00
85827eddc5 Fixed footers 2025-03-28 18:26:14 +00:00
a6a62e84d3 Merge pull request 'Nouveau footer - changement de nom des images' (#3) from wip into main
Reviewed-on: #3
2025-03-28 19:22:54 +01:00
06eef00c79 Nouveau footer - changement de nom des images 2025-03-28 18:20:37 +00:00
6d4a160416 New logo - dark 2025-03-28 16:10:40 +00:00
7894b03883 Fixed schematics definition 2025-03-28 15:26:25 +00:00
c787c61332 Merge branch 'wip' 2025-03-28 15:24:24 +00:00
d6b5273e5a Fixed list 2025-03-28 15:23:55 +00:00
10ab4f6fc0 Merge branch 'wip' 2025-03-28 15:21:02 +00:00
f133344f7a fixed schematics 2025-03-28 15:20:50 +00:00
d1609d0776 Contributeurs 2025-03-28 15:05:18 +00:00
bf5fa6203b Merge branch 'wip' 2025-03-28 14:37:12 +00:00
aae087c91b New pp 2025-03-28 14:36:48 +00:00
b69b45a03f Merge branch 'wip' 2025-03-28 13:36:33 +00:00
5503fbf275 fixed logo 2025-03-28 13:35:56 +00:00
2b192247bb Merge pull request 'Nouvelle pp + section capture' (#2) from wip into main
Reviewed-on: #2
2025-03-28 10:16:27 +01:00
e1e3502be2 Nouvelle pp + section capture 2025-03-28 09:12:55 +00:00
e585f18a93 Merge branch 'wip' 2025-03-27 22:18:18 +00:00
fc6ae91c55 Minor changes 2025-03-27 22:17:30 +00:00
a496a1b1bf Merge branch 'wip' 2025-03-27 22:09:03 +00:00
1722f276a9 Logo 2025-03-27 22:07:26 +00:00
38b60300c4 Logo 2025-03-27 22:05:59 +00:00
6a6336a07c Merge pull request 'wip - Fusion refonte README.md & LICENSE' (#1) from wip into main
Reviewed-on: #1
2025-03-27 21:01:19 +01:00
ef27e320e2 Refonte README + License 2025-03-27 19:57:30 +00:00
16e80db82f Merge branch 'wip' of https://git.djeex.fr/Djeex/nvidia-stock-bot into wip 2025-03-27 18:31:07 +00:00
3dd9dd5946 Minor changes 2025-03-27 18:30:07 +00:00
9dd3d76d7f Disclaimer IA 2025-03-27 18:29:49 +00:00
45b6bed116 Minor changes 2025-03-27 18:27:03 +00:00
24cad015bc Minor changes 2025-03-27 18:25:55 +00:00
b0309f34fe Disclaimer IA 2025-03-27 18:23:53 +00:00
841ffb5efe Minor changes 2025-03-27 10:59:14 +00:00
d4e34f20ec Minor changes 2025-03-27 10:39:43 +00:00
ddfbe853e9 fix png -> jpg 2025-03-27 10:19:06 +00:00
4115825953 illustration discord 2025-03-27 10:18:18 +00:00
dfa3a5e19c In english 2025-03-26 15:38:46 +00:00
cdbb1be864 Minor change 2025-03-26 12:01:31 +00:00
3ed38e7aa3 badge license 2025-03-26 12:01:12 +00:00
c392efce35 Ajout license 2025-03-26 11:55:09 +00:00
2761a9dacb License 2025-03-26 11:53:23 +00:00
006531aacc Maj intro 2025-03-14 15:12:40 +00:00
a8d4c05c81 Fix python section 2025-03-14 15:08:30 +00:00
280e8d0347 Maj schema 2025-03-14 15:05:06 +00:00
53c46b1d2f WIP to MAIN 2025-03-14 12:02:54 +00:00
5b1e180c6f Minor changes 2025-03-14 11:51:48 +00:00
7872f8aa00 Fixed sku notif 2025-03-14 11:33:50 +00:00
23b2f375fc Minor fixes 2025-03-14 11:21:56 +00:00
62bc725a1c Fixed coma 2025-03-14 11:04:57 +00:00
b381efd257 Readme 2025-03-14 10:55:41 +00:00
037f5bebf8 SKU changes & better notif 2025-03-14 10:53:59 +00:00
0803e3074b Meilleur masquage du webhook 2025-02-24 09:58:36 +00:00
21f6bd7a1f Better notification 2025-02-21 22:39:23 +00:00
8e2e91a760 pictures 2025-02-21 21:14:09 +00:00
d95afc4627 Commentaires 2025-02-21 16:08:10 +00:00
d97a32623f Illustration 2025-02-21 15:55:23 +00:00
705a96d0c2 Minor changes 2025-02-21 15:24:13 +00:00
66ca9e907d Merge branch 'main' of https://git.djeex.fr/Djeex/nvidia-stock-bot 2025-02-21 15:20:18 +00:00
82557b77ab Last commit 2025-02-21 15:13:09 +00:00
d48d9d537b auto sku 2025-02-19 17:25:23 +00:00
8d1fd5d94a auto sku 2025-02-19 17:23:48 +00:00
86665b7137 auto sku 2025-02-19 17:19:49 +00:00
8ccc925966 Readme 2025-02-11 21:04:49 +00:00
14 changed files with 1013 additions and 238 deletions

View File

@ -1,22 +1,18 @@
# Utiliser une image de base légère de Python (ici Python 3.9)
FROM python:3.9-slim
FROM python:3.11-alpine
RUN apk add --no-cache ca-certificates
# Définir le répertoire de travail à l'intérieur du conteneur
WORKDIR /app
# Copier le script Python dans le répertoire de travail
COPY nvidia-stock-bot.py /app/
COPY app/nvidia-stock-bot.py /app/
# Copier un éventuel fichier requirements.txt pour installer des dépendances
# Si des dépendances supplémentaires sont nécessaires, ajoutez-les dans requirements.txt
COPY requirements.txt /app/
COPY app/requirements.txt /app/
COPY app/localization.json /app/
# Installer les dépendances Python à partir de requirements.txt
RUN pip install --no-cache-dir -r requirements.txt
# Définir les variables d'environnement par défaut (modifiable lors du lancement du conteneur)
ENV DISCORD_WEBHOOK_URL="https://example.com/webhook" \
REFRESH_TIME="60"
REFRESH_TIME="30"
# Exposer un point de commande pour exécuter le script
CMD ["python", "nvidia-stock-bot.py"]

160
LICENSE Normal file
View File

@ -0,0 +1,160 @@
Creative Commons Attribution-NonCommercial 4.0 International
Copyright (c) Djeex - 2025
Creative Commons Corporation (“Creative Commons”) is not a law firm and does not provide legal services or legal advice. Distribution of Creative Commons public licenses does not create a lawyer-client or other relationship. Creative Commons makes its licenses and related information available on an “as-is” basis. Creative Commons gives no warranties regarding its licenses, any material licensed under their terms and conditions, or any related information. Creative Commons disclaims all liability for damages resulting from their use to the fullest extent possible.
Using Creative Commons Public Licenses
Creative Commons public licenses provide a standard set of terms and conditions that creators and other rights holders may use to share original works of authorship and other material subject to copyright and certain other rights specified in the public license below. The following considerations are for informational purposes only, are not exhaustive, and do not form part of our licenses.
Considerations for licensors: Our public licenses are intended for use by those authorized to give the public permission to use material in ways otherwise restricted by copyright and certain other rights. Our licenses are irrevocable. Licensors should read and understand the terms and conditions of the license they choose before applying it. Licensors should also secure all rights necessary before applying our licenses so that the public can reuse the material as expected. Licensors should clearly mark any material not subject to the license. This includes other CC-licensed material, or material used under an exception or limitation to copyright. More considerations for licensors.
Considerations for the public: By using one of our public licenses, a licensor grants the public permission to use the licensed material under specified terms and conditions. If the licensors permission is not necessary for any reasonfor example, because of any applicable exception or limitation to copyrightthen that use is not regulated by the license. Our licenses grant only permissions under copyright and certain other rights that a licensor has authority to grant. Use of the licensed material may still be restricted for other reasons, including because others have copyright or other rights in the material. A licensor may make special requests, such as asking that all changes be marked or described. Although not required by our licenses, you are encouraged to respect those requests where reasonable. More considerations for the public.
Creative Commons Attribution-NonCommercial 4.0 International Public License
By exercising the Licensed Rights (defined below), You accept and agree to be bound by the terms and conditions of this Creative Commons Attribution-NonCommercial 4.0 International Public License ("Public License"). To the extent this Public License may be interpreted as a contract, You are granted the Licensed Rights in consideration of Your acceptance of these terms and conditions, and the Licensor grants You such rights in consideration of benefits the Licensor receives from making the Licensed Material available under these terms and conditions.
Section 1 Definitions.
a. Adapted Material means material subject to Copyright and Similar Rights that is derived from or based upon the Licensed Material and in which the Licensed Material is translated, altered, arranged, transformed, or otherwise modified in a manner requiring permission under the Copyright and Similar Rights held by the Licensor. For purposes of this Public License, where the Licensed Material is a musical work, performance, or sound recording, Adapted Material is always produced where the Licensed Material is synched in timed relation with a moving image.
b. Adapter's License means the license You apply to Your Copyright and Similar Rights in Your contributions to Adapted Material in accordance with the terms and conditions of this Public License.
c. Copyright and Similar Rights means copyright and/or similar rights closely related to copyright including, without limitation, performance, broadcast, sound recording, and Sui Generis Database Rights, without regard to how the rights are labeled or categorized. For purposes of this Public License, the rights specified in Section 2(b)(1)-(2) are not Copyright and Similar Rights.
d. Effective Technological Measures means those measures that, in the absence of proper authority, may not be circumvented under laws fulfilling obligations under Article 11 of the WIPO Copyright Treaty adopted on December 20, 1996, and/or similar international agreements.
e. Exceptions and Limitations means fair use, fair dealing, and/or any other exception or limitation to Copyright and Similar Rights that applies to Your use of the Licensed Material.
f. Licensed Material means the artistic or literary work, database, or other material to which the Licensor applied this Public License.
g. Licensed Rights means the rights granted to You subject to the terms and conditions of this Public License, which are limited to all Copyright and Similar Rights that apply to Your use of the Licensed Material and that the Licensor has authority to license.
h. Licensor means the individual(s) or entity(ies) granting rights under this Public License.
i. NonCommercial means not primarily intended for or directed towards commercial advantage or monetary compensation. For purposes of this Public License, the exchange of the Licensed Material for other material subject to Copyright and Similar Rights by digital file-sharing or similar means is NonCommercial provided there is no payment of monetary compensation in connection with the exchange.
j. Share means to provide material to the public by any means or process that requires permission under the Licensed Rights, such as reproduction, public display, public performance, distribution, dissemination, communication, or importation, and to make material available to the public including in ways that members of the public may access the material from a place and at a time individually chosen by them.
k. Sui Generis Database Rights means rights other than copyright resulting from Directive 96/9/EC of the European Parliament and of the Council of 11 March 1996 on the legal protection of databases, as amended and/or succeeded, as well as other essentially equivalent rights anywhere in the world.
l. You means the individual or entity exercising the Licensed Rights under this Public License. Your has a corresponding meaning.
Section 2 Scope.
a. License grant.
1. Subject to the terms and conditions of this Public License, the Licensor hereby grants You a worldwide, royalty-free, non-sublicensable, non-exclusive, irrevocable license to exercise the Licensed Rights in the Licensed Material to:
A. reproduce and Share the Licensed Material, in whole or in part, for NonCommercial purposes only; and
B. produce, reproduce, and Share Adapted Material for NonCommercial purposes only.
2. Exceptions and Limitations. For the avoidance of doubt, where Exceptions and Limitations apply to Your use, this Public License does not apply, and You do not need to comply with its terms and conditions.
3. Term. The term of this Public License is specified in Section 6(a).
4. Media and formats; technical modifications allowed. The Licensor authorizes You to exercise the Licensed Rights in all media and formats whether now known or hereafter created, and to make technical modifications necessary to do so. The Licensor waives and/or agrees not to assert any right or authority to forbid You from making technical modifications necessary to exercise the Licensed Rights, including technical modifications necessary to circumvent Effective Technological Measures. For purposes of this Public License, simply making modifications authorized by this Section 2(a)(4) never produces Adapted Material.
5. Downstream recipients.
A. Offer from the Licensor Licensed Material. Every recipient of the Licensed Material automatically receives an offer from the Licensor to exercise the Licensed Rights under the terms and conditions of this Public License.
B. No downstream restrictions. You may not offer or impose any additional or different terms or conditions on, or apply any Effective Technological Measures to, the Licensed Material if doing so restricts exercise of the Licensed Rights by any recipient of the Licensed Material.
6. No endorsement. Nothing in this Public License constitutes or may be construed as permission to assert or imply that You are, or that Your use of the Licensed Material is, connected with, or sponsored, endorsed, or granted official status by, the Licensor or others designated to receive attribution as provided in Section 3(a)(1)(A)(i).
b. Other rights.
1. Moral rights, such as the right of integrity, are not licensed under this Public License, nor are publicity, privacy, and/or other similar personality rights; however, to the extent possible, the Licensor waives and/or agrees not to assert any such rights held by the Licensor to the limited extent necessary to allow You to exercise the Licensed Rights, but not otherwise.
2. Patent and trademark rights are not licensed under this Public License.
3. To the extent possible, the Licensor waives any right to collect royalties from You for the exercise of the Licensed Rights, whether directly or through a collecting society under any voluntary or waivable statutory or compulsory licensing scheme. In all other cases the Licensor expressly reserves any right to collect such royalties, including when the Licensed Material is used other than for NonCommercial purposes.
Section 3 License Conditions.
Your exercise of the Licensed Rights is expressly made subject to the following conditions.
a. Attribution.
1. If You Share the Licensed Material (including in modified form), You must:
A. retain the following if it is supplied by the Licensor with the Licensed Material:
i. identification of the creator(s) of the Licensed Material and any others designated to receive attribution, in any reasonable manner requested by the Licensor (including by pseudonym if designated);
ii. a copyright notice;
iii. a notice that refers to this Public License;
iv. a notice that refers to the disclaimer of warranties;
v. a URI or hyperlink to the Licensed Material to the extent reasonably practicable;
B. indicate if You modified the Licensed Material and retain an indication of any previous modifications; and
C. indicate the Licensed Material is licensed under this Public License, and include the text of, or the URI or hyperlink to, this Public License.
2. You may satisfy the conditions in Section 3(a)(1) in any reasonable manner based on the medium, means, and context in which You Share the Licensed Material. For example, it may be reasonable to satisfy the conditions by providing a URI or hyperlink to a resource that includes the required information.
3. If requested by the Licensor, You must remove any of the information required by Section 3(a)(1)(A) to the extent reasonably practicable.
4. If You Share Adapted Material You produce, the Adapter's License You apply must not prevent recipients of the Adapted Material from complying with this Public License.
Section 4 Sui Generis Database Rights.
Where the Licensed Rights include Sui Generis Database Rights that apply to Your use of the Licensed Material:
a. for the avoidance of doubt, Section 2(a)(1) grants You the right to extract, reuse, reproduce, and Share all or a substantial portion of the contents of the database for NonCommercial purposes only;
b. if You include all or a substantial portion of the database contents in a database in which You have Sui Generis Database Rights, then the database in which You have Sui Generis Database Rights (but not its individual contents) is Adapted Material; and
c. You must comply with the conditions in Section 3(a) if You Share all or a substantial portion of the contents of the database.
For the avoidance of doubt, this Section 4 supplements and does not replace Your obligations under this Public License where the Licensed Rights include other Copyright and Similar Rights.
Section 5 Disclaimer of Warranties and Limitation of Liability.
a. Unless otherwise separately undertaken by the Licensor, to the extent possible, the Licensor offers the Licensed Material as-is and as-available, and makes no representations or warranties of any kind concerning the Licensed Material, whether express, implied, statutory, or other. This includes, without limitation, warranties of title, merchantability, fitness for a particular purpose, non-infringement, absence of latent or other defects, accuracy, or the presence or absence of errors, whether or not known or discoverable. Where disclaimers of warranties are not allowed in full or in part, this disclaimer may not apply to You.
b. To the extent possible, in no event will the Licensor be liable to You on any legal theory (including, without limitation, negligence) or otherwise for any direct, special, indirect, incidental, consequential, punitive, exemplary, or other losses, costs, expenses, or damages arising out of this Public License or use of the Licensed Material, even if the Licensor has been advised of the possibility of such losses, costs, expenses, or damages. Where a limitation of liability is not allowed in full or in part, this limitation may not apply to You.
c. The disclaimer of warranties and limitation of liability provided above shall be interpreted in a manner that, to the extent possible, most closely approximates an absolute disclaimer and waiver of all liability.
Section 6 Term and Termination.
a. This Public License applies for the term of the Copyright and Similar Rights licensed here. However, if You fail to comply with this Public License, then Your rights under this Public License terminate automatically.
b. Where Your right to use the Licensed Material has terminated under Section 6(a), it reinstates:
1. automatically as of the date the violation is cured, provided it is cured within 30 days of Your discovery of the violation; or
2. upon express reinstatement by the Licensor.
For the avoidance of doubt, this Section 6(b) does not affect any right the Licensor may have to seek remedies for Your violations of this Public License.
c. For the avoidance of doubt, the Licensor may also offer the Licensed Material under separate terms or conditions or stop distributing the Licensed Material at any time; however, doing so will not terminate this Public License.
d. Sections 1, 5, 6, 7, and 8 survive termination of this Public License.
Section 7 Other Terms and Conditions.
a. The Licensor shall not be bound by any additional or different terms or conditions communicated by You unless expressly agreed.
b. Any arrangements, understandings, or agreements regarding the Licensed Material not stated herein are separate from and independent of the terms and conditions of this Public License.
Section 8 Interpretation.
a. For the avoidance of doubt, this Public License does not, and shall not be interpreted to, reduce, limit, restrict, or impose conditions on any use of the Licensed Material that could lawfully be made without permission under this Public License.
b. To the extent possible, if any provision of this Public License is deemed unenforceable, it shall be automatically reformed to the minimum extent necessary to make it enforceable. If the provision cannot be reformed, it shall be severed from this Public License without affecting the enforceability of the remaining terms and conditions.
c. No term or condition of this Public License will be waived and no failure to comply consented to unless expressly agreed to by the Licensor.
d. Nothing in this Public License constitutes or may be interpreted as a limitation upon, or waiver of, any privileges and immunities that apply to the Licensor or You, including from the legal processes of any jurisdiction or authority.
Creative Commons is not a party to its public licenses. Notwithstanding, Creative Commons may elect to apply one of its public licenses to material it publishes and in those instances will be considered the “Licensor.” Except for the limited purpose of indicating that material is shared under a Creative Commons public license or as otherwise permitted by the Creative Commons policies published at creativecommons.org/policies, Creative Commons does not authorize the use of the trademark “Creative Commons” or any other trademark or logo of Creative Commons without its prior written consent including, without limitation, in connection with any unauthorized modifications to any of its public licenses or any other arrangements, understandings, or agreements concerning use of licensed material. For the avoidance of doubt, this paragraph does not form part of the public licenses.
Creative Commons may be contacted at creativecommons.org.

222
README.md
View File

@ -1,110 +1,190 @@
# Nvidia Stock Bot - WIP -
Par KevOut & Djeex
<h1 align="center">Nvidia Stock Bot</h1>
<div align="center">
<a href="https://creativecommons.org/licenses/by-nc/4.0/" target="_blank">
<img src="https://img.shields.io/badge/License-CC%20BY--NC%204.0-8E44AD?style=flat-square" alt="License: CC BY-NC 4.0">
</a>
</div>
<div align="center" >
<img src="https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/nvidia-stock-bot-logo.png" alt="Nvidia Stock Bot" width="300">
</div>
[![](https://img.shields.io/badge/JV%20hardware-rejoindre-green?style=flat-square&logo=discord&logoColor=%23fff&label=JV%20hardware&link=https%3A%2F%2Fdiscord.gg%2Fgxffg3GA96)](https://discord.gg/gxffg3GA96)
**🤖 Nvidia Stock Bot** - A bot that alerts you in real-time about **Nvidia RTX FE** GPU stock availability through Discord notifications.
> [!NOTE]
>_The code was partially written and structured using a generative AI._
>
>_Github repo is a mirror of https://git.djeex.fr/Djeex/nvidia-stock-bot. You'll find full package, history and release note there._
Ce robot :
- Appelle régulièrement l'api des stocks français de nvidia FE (par défaut toutes les 60s)
- Vérifie si RTX 5090, RTX 5080, RTX 5070ti et RTX 5070 sont en stock
- Si du stock est trouvé, envoie une notification discord via le webhook paramétré
## 📌 Table of Contents
<img src="https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/nvbot.png" align="center">
- [✨ Features](#-features)
- [🐳 Docker Installation without cloning the repo (quick)](#-docker-installation-without-the-repo-quick)
- [🐙 Docker Installation with the repo (developer)](#-docker-installation-with-the-repo)
- [🐍 Python Installation (developer)](#-python-installation)
- [🖼️ Screenshots](#-screenshots)
- [🐞 Common issues](#-common-issues)
- [🧑‍💻 Contributors](#-contributors)
## ✨ Features
Trois modes d'installation :
- [Avec le dépot Git et Docker](https://git.djeex.fr/Djeex/nvidia-stock-bot/#installation-avec-le-d%C3%A9pot)
- [Sans le dépot Git et avec notre image docker fournie](https://git.djeex.fr/Djeex/nvidia-stock-bot/#installation-sans-le-d%C3%A9pot-avec-docker-compose)
- [Avec python (développeurs)](https://git.djeex.fr/Djeex/nvidia-stock-bot/#installation-sans-le-d%C3%A9pot-avec-docker-compose)
- Selectable GPU models
- Discord `@everyone` or specified role on SKU change (possible imminent drop)
- Discord `@everyone` or specified role notification when stock is detected, including model, price, and link
- Silent Discord notification when no stock is detected
- Selectable notification language
- Selectable notification server name in footer
- Selectable check frequency
## Installation avec le dépot
## 🐳 Docker Installation without the repo (quick)
Vous trouverez-ci dessous les instructions pour installer le dépot, compiler l'image docker, et lancer le conteneur. Avec cette solution, votre bot tournera tout seul tant que le conteneur est actif.
Below are the instructions to set up the container using our pre-built image. With this setup, your bot will run independently as long as the container is active.
### Pré-requis
- Git
- Docker
**Requirements**
- [Docker](https://docs.docker.com/engine/install/)
### Cloner et paramétrer
**Configuration**
Clonez le repo :
```sh
git clone https://git.djeex.fr/Djeex/nvidia-stock-bot.git
- Create a folder named `nvidia-stock-bot`
- Create a `compose.yaml` file inside that folder with the following content:
```yaml
services:
nvidia-stock-bot:
image: git.djeex.fr/djeex/nvidia-stock-bot:latest
container_name: nvidia-stock-bot
restart: unless-stopped
environment:
# Minimal environment variables
- PRODUCT_NAMES= # Exact GPU name (e.g. "RTX 5080, RTX 5090")
- DISCORD_WEBHOOK_URL= # Your Discord webhook URL
- API_URL_SKU= # API listing the product for your country
- API_URL_STOCK= # API providing stock data for your country
- PRODUCT_URL= # GPU purchase URL for your country
- PYTHONUNBUFFERED=1 # Enables real-time log output
command: python nvidia-stock-bot.py
```
Rendez vous dans le dossier `nvidia-stock-bot` et compilez l'image docker :
```sh
docker build -t nvidia-stock-bot .
```
**Environment Variables:**
Rendez-vous dans le dossier `nvidia-stock-bot/docker` et éditez le fichier `.env` avec :
- l'url de votre webhook discord
- la fréquence de consultation des stock (par défaut 60s, attention à ne pas trop descendre sous peine de blocage de votre adresse IP par nVidia)
| Variable | Description | Possible Values | Default Value |
|---------------------|-------------------------------------------------|------------------------------------------------------------------|-------------------------------------------------------------------------------------------------------------|
| `PRODUCT_NAMES` | The exact GPU names you're searching for | `RTX 5080, RTX 5090` | |
| `DISCORD_WEBHOOK_URL` | Your Discord webhook URL | A valid URL | |
| `DISCORD_SERVER_NAME` | The name of your server, displayed in notification's footer | A text | Shared for free |
| `DISCORD_NOTIFICATION_LANGUAGE` | Your language for notification's content | `bg`, `cs`, `da`, `de`, `el`, `en`, `es`, `et`, `fi`, `fr`, `ga`, `hr`, `hu`, `it`, `lt`, `lv`, `mt`, `nl`, `pl`, `pt`, `ro`, `sk`, `sl`, `sv` | `en` |
| `DISCORD_ROLES` | List of Discord roles ID in the same order than `PRODUCT_NAMES` values, found in your discord server settings (with user profile developer mode enabled) | `<@&12345><@&6789>` | @everyone |
| `REFRESH_TIME` | Script refresh interval in seconds | `60`, `30`, etc. | `30` |
| `API_URL_SKU` | API listing the product | A URL. API url can change over time. For now, you can use the default one and change the `locale` parameter to yours (e.g. `locale=en-gb`) | `https://api.nvidia.partners/edge/product/search?page=1&limit=100&locale=fr-fr&Manufacturer=Nvidia` |
| `API_URL_STOCK` | API providing stock data | A URL. API url can change over time. For now, you can use the default one and change the `locale` parameter to yours (e.g. `locale=en-gb`) | `https://api.store.nvidia.com/partner/v1/feinventory?locale=fr-fr&skus=` |
| `PRODUCT_URL` | GPU purchase URL. There isn't any direct link workinf right now, so put the generic marketplace url listing all FE products | A URL. API url can change over time. For now, you can use the default one and change the locale parameter to yours (e.g. `/en-gb/`) | `https://marketplace.nvidia.com/fr-fr/consumer/graphics-cards/?locale=fr-fr&page=1&limit=12&manufacturer=NVIDIA` |
| `TEST_MODE` | For testing without sending notifications | `True`, `False` | `False` |
| `PYTHONUNBUFFERED` | Enables real-time log output | `1`, `0` | `1` |
### Lancer l'image
**Run the image**
Rendez-vous dans le dossier `nvidia-stock-bot/docker` et lancez le conteneur :
Navigate to the `nvidia-stock-bot` folder and launch the container:
```sh
docker compose up -d
```
### Voir les logs pour vérifier le bon fonctionnement
**Check logs to verify operation**
```sh
docker logs -f nvidia-stock-bot
```
## Installation sans le dépot avec docker compose
## 📦 Docker Installation with the repo
Vous trouverez-ci dessous les instructions pour configurer le conteneur avec notre image pré-compilée. Avec cette solution, votre bot tournera tout seul tant que le conteneur est actif.
Instructions below show how to install the repo, build the Docker image, and launch the container. Your bot will run independently as long as the container is active.
### Pré-requis
- Docker
**Requirements**
- [Git](https://git-scm.com/docs)
- [Docker](https://docs.docker.com/engine/install/)
### Configuration
**Clone and configure**
```yaml
version: "3.8"
services:
nvidia-stock-bot:
image: git.djeex.fr/djeex/nvidia-stock-bot:wip
container_name: nvidia-stock-bot
restart: always # Le conteneur redémarrera automatiquement en cas d'échec
environment:
- DISCORD_WEBHOOK_URL= # URL de votre webhook Discord
- REFRESH_TIME= # Durée de rafraichissement du script en secondes
- GPU_TARGETS= #SKU
- API_URL= #URL de l'API
- PYTHONUNBUFFERED=1 # Permet d'afficher les logs en temps réel
command: python nvidia-stock-bot.py # Lance le script Python au démarrage du conteneur
- Clone the repo:
```sh
git clone https://git.djeex.fr/Djeex/nvidia-stock-bot.git
```
## Installation avec Python
- Navigate to `nvidia-stock-bot` and build the Docker image:
```sh
docker build -t nvidia-stock-bot .
```
Vous trouverez ci-dessous comment exécuter directement le script Python. Avec cette solution, le bot s'arretera si vous fermez votre terminal.
- Then go to `nvidia-stock-bot/docker` and edit the `.env` file with:
- Your Discord webhook URL
- The API and product URLs
- Stock checking frequency (default: 60s; lowering too much may get your IP blocked by Nvidia)
### Pré-requis
**Run the image**
- Python 3.11 ou plus
- requests : `pip install requests`
Navigate to `nvidia-stock-bot/docker` and launch the container:
```sh
docker compose up -d
```
### Configuration
**Check logs to verify operation**
- Créez un environnement virtuel (exemple : `python3 -m venv nom_de_l_environnement` )
- Créez un dossier et aller dedans
- Téléchargez le script python :
```sh
docker logs -f nvidia-stock-bot
```
```sh
curl -o nvidia-stock-bot.py -# https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/nvidia-stock-bot.py
```
- Exportez les variables d'environnement avec votre webhook discord et le temps de rafraichissement en secondes, par exemple :
## 🐍 Python Installation
```sh
export DISCORD_WEBHOOK_URL="https://votre_url_discord"
export REFRESH_TIME="60"
```
- Lancez le script
Instructions to directly run the Python script. Note: the bot stops when you close the terminal.
```sh
python nvidia-stock-bot.py
```
**Requirements**
- Python 3.11 or newer
- requests: `pip install requests`
**Configuration**
- Clone the repo:
```sh
git clone https://git.djeex.fr/Djeex/nvidia-stock-bot.git
```
- Navigate to `nvidia-stock-bot` and create a virtual environment (e.g., `python3 -m venv env_name`)
- Export the environment variables with your webhook and refresh time, for exemple:
```sh
export DISCORD_WEBHOOK_URL="https://your_discord_url"
export PRODUCT_NAMES=RTX 5080, RTX 5090
export DISCORD_ROLES=<@&12345>, <@&6789>
export REFRESH_TIME="60"
export API_URL_SKU="https://api.nvidia.partners/edge/product/search?page=1&limit=100&locale=fr-fr&Manufacturer=Nvidia&gpu=RTX%205080"
export API_URL_STOCK="https://api.store.nvidia.com/partner/v1/feinventory?locale=fr-fr&skus="
export PRODUCT_URL="https://marketplace.nvidia.com/fr-fr/consumer/graphics-cards/?locale=fr-fr&page=1&limit=12&gpu=RTX%205080&manufacturer=NVIDIA"
export TEST_MODE=false
export PYTHONUNBUFFERED=1
```
- Run the script
```sh
python nvidia-stock-bot.py
```
## 🖼️ Screenshots
<div align="center" >
<img src="https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/nvidia-stock-bot-discord.png" alt="Nvidia Stock Bot - screenshots">
</div>
## 🐞 Common issues
Error when trying to reach product API url :
- `API_SKU_URL` may be wrong
- Your IP may be blacklisted by nvidia. Try to use a VPN.
- nvidia API may be down
## 🧑‍💻 Contributors
Thanks for their contributions:
- Djeex
- KevOut
- Extreme2pac

266
app/localization.json Normal file
View File

@ -0,0 +1,266 @@
{
"en": {
"in_stock_title": "🚀 {gpu_name} IN STOCK!",
"out_of_stock_title": "❌ {gpu_name} is out of stock",
"sku_change_title": "🔄 SKU change detected for {gpu_name}",
"buy_now": "**:point_right: [Buy now]({product_link})**",
"price": "Price",
"time": "Time",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Old SKU** : `{old_sku}`\n**New SKU** : `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Possible imminent drop!"
},
"bg": {
"in_stock_title": "🚀 {gpu_name} ИМА НАЛИЧНОСТ!",
"out_of_stock_title": "❌ {gpu_name} е изчерпан",
"sku_change_title": "🔄 Засечена промяна на SKU за {gpu_name}",
"buy_now": "**:point_right: [Купи сега]({product_link})**",
"price": "Цена",
"time": "Час",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Стар SKU**: `{old_sku}`\n**Нов SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Възможен наближаващ спад!"
},
"cs": {
"in_stock_title": "🚀 {gpu_name} SKLADEM!",
"out_of_stock_title": "❌ {gpu_name} je vyprodán",
"sku_change_title": "🔄 Změna SKU nalezena pro {gpu_name}",
"buy_now": "**:point_right: [Koupit nyní]({product_link})**",
"price": "Cena",
"time": "Čas",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Starý SKU**: `{old_sku}`\n**Nový SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Možný blížící se pokles!"
},
"da": {
"in_stock_title": "🚀 {gpu_name} PÅ LAGER!",
"out_of_stock_title": "❌ {gpu_name} er udsolgt",
"sku_change_title": "🔄 SKU-ændring fundet for {gpu_name}",
"buy_now": "**:point_right: [Køb nu]({product_link})**",
"price": "Pris",
"time": "Tid",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Gammelt SKU**: `{old_sku}`\n**Nyt SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Muligt snarligt drop!"
},
"de": {
"in_stock_title": "🚀 {gpu_name} AUF LAGER!",
"out_of_stock_title": "❌ {gpu_name} ist ausverkauft",
"sku_change_title": "🔄 SKU-Änderung erkannt für {gpu_name}",
"buy_now": "**:point_right: [Jetzt kaufen]({product_link})**",
"price": "Preis",
"time": "Zeit",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Alte SKU**: `{old_sku}`\n**Neue SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Möglicher bevorstehender Drop!"
},
"el": {
"in_stock_title": "🚀 {gpu_name} ΔΙΑΘΕΣΙΜΟ!",
"out_of_stock_title": "❌ {gpu_name} εξαντλήθηκε",
"sku_change_title": "🔄 Ανίχνευση αλλαγής SKU για {gpu_name}",
"buy_now": "**:point_right: [Αγόρασε τώρα]({product_link})**",
"price": "Τιμή",
"time": "Ώρα",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Παλαιό SKU**: `{old_sku}`\n**Νέο SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Πιθανή επικείμενη πτώση!"
},
"es": {
"in_stock_title": "🚀 {gpu_name} ¡EN STOCK!",
"out_of_stock_title": "❌ {gpu_name} está agotado",
"sku_change_title": "🔄 Cambio de SKU detectado para {gpu_name}",
"buy_now": "**:point_right: [Comprar ahora]({product_link})**",
"price": "Precio",
"time": "Hora",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**SKU antiguo**: `{old_sku}`\n**SKU nuevo**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ ¡Posible drop inminente!"
},
"et": {
"in_stock_title": "🚀 {gpu_name} LAOS!",
"out_of_stock_title": "❌ {gpu_name} on välja müüdud",
"sku_change_title": "🔄 Tuvastatud SKU muudatus {gpu_name}",
"buy_now": "**:point_right: [Osta kohe]({product_link})**",
"price": "Hind",
"time": "Aeg",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Vana SKU**: `{old_sku}`\n**Uus SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Võimalik varsti tulemas drop!"
},
"fi": {
"in_stock_title": "🚀 {gpu_name} VARASTOSSA!",
"out_of_stock_title": "❌ {gpu_name} on loppunut",
"sku_change_title": "🔄 SKU-muutos havaittu {gpu_name}",
"buy_now": "**:point_right: [Osta nyt]({product_link})**",
"price": "Hinta",
"time": "Aika",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Vanha SKU**: `{old_sku}`\n**Uusi SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Mahdollinen lähestyvä drop!"
},
"fr": {
"in_stock_title": "🚀 {gpu_name} EN STOCK !",
"out_of_stock_title": "❌ {gpu_name} est en rupture",
"sku_change_title": "🔄 Changement de SKU détecté pour {gpu_name}",
"buy_now": "**:point_right: [Acheter maintenant]({product_link})**",
"price": "Prix",
"time": "Heure",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Ancien SKU** : `{old_sku}`\n**Nouveau SKU** : `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Éventuel drop imminent!"
},
"hr": {
"in_stock_title": "🚀 {gpu_name} NA ZALIHI!",
"out_of_stock_title": "❌ {gpu_name} je rasprodan",
"sku_change_title": "🔄 Promjena SKU za {gpu_name}",
"buy_now": "**:point_right: [Kupi sada]({product_link})**",
"price": "Cijena",
"time": "Vrijeme",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Stari SKU**: `{old_sku}`\n**Novi SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Mogući nadolazeći drop!"
},
"hu": {
"in_stock_title": "🚀 {gpu_name} RAKTÁRON!",
"out_of_stock_title": "❌ {gpu_name} elfogyott",
"sku_change_title": "🔄 SKU változás észlelve: {gpu_name}",
"buy_now": "**:point_right: [Vásárlás most]({product_link})**",
"price": "Ár",
"time": "Idő",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Régi SKU**: `{old_sku}`\n**Új SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Lehetséges közelgő drop!"
},
"ga": {
"in_stock_title": "🚀 {gpu_name} I STOC!",
"out_of_stock_title": "❌ níl {gpu_name} ar fáil",
"sku_change_title": "🔄 SKU athraithe aimsithe do {gpu_name}",
"buy_now": "**:point_right: [Ceannaigh anois]({product_link})**",
"price": "Praghas",
"time": "Am",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Sean SKU**: `{old_sku}`\n**SKU Nua**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ I bhfad is gaire drop féideartha!"
},
"it": {
"in_stock_title": "🚀 {gpu_name} DISPONIBILE!",
"out_of_stock_title": "❌ {gpu_name} esaurito",
"sku_change_title": "🔄 Modifica SKU rilevata per {gpu_name}",
"buy_now": "**:point_right: [Acquista ora]({product_link})**",
"price": "Prezzo",
"time": "Ora",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Vecchio SKU**: `{old_sku}`\n**Nuovo SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Possibile drop imminente!"
},
"lv": {
"in_stock_title": "🚀 {gpu_name} KRĀJUMĀ!",
"out_of_stock_title": "❌ {gpu_name} nav pieejams",
"sku_change_title": "🔄 SKU izmaiņas: {gpu_name}",
"buy_now": "**:point_right: [Pirkt tagad]({product_link})**",
"price": "Cena",
"time": "Laiks",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Vecais SKU**: `{old_sku}`\n**Jaunais SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Iespējams gaidāms drop!"
},
"lt": {
"in_stock_title": "🚀 {gpu_name} SANDĖLYJE!",
"out_of_stock_title": "❌ {gpu_name} išparduotas",
"sku_change_title": "🔄 SKU pakeitimas aptiktas: {gpu_name}",
"buy_now": "**:point_right: [Pirkti dabar]({product_link})**",
"price": "Kaina",
"time": "Laikas",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Senas SKU**: `{old_sku}`\n**Naujas SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Rugsėjo drop'as gali artėti!"
},
"mt": {
"in_stock_title": "🚀 {gpu_name} F'ĠESTA!",
"out_of_stock_title": "❌ {gpu_name} mhux disponibbli",
"sku_change_title": "🔄 Bidla SKU skoperta għal {gpu_name}",
"buy_now": "**:point_right: [Ixtri issa]({product_link})**",
"price": "Prezz",
"time": "Ħin",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**SKU Antik**: `{old_sku}`\n**SKU Ġdid**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Possibbli drop imminenti!"
},
"nl": {
"in_stock_title": "🚀 {gpu_name} OP VOORRAAD!",
"out_of_stock_title": "❌ {gpu_name} is uitverkocht",
"sku_change_title": "🔄 SKU-wijziging gedetecteerd voor {gpu_name}",
"buy_now": "**:point_right: [Nu kopen]({product_link})**",
"price": "Prijs",
"time": "Tijd",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Oude SKU**: `{old_sku}`\n**Nieuwe SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Mogelijke aanstaande drop!"
},
"pl": {
"in_stock_title": "🚀 {gpu_name} DOSTĘPNE!",
"out_of_stock_title": "❌ {gpu_name} jest niedostępny",
"sku_change_title": "🔄 Wykryto zmianę SKU dla {gpu_name}",
"buy_now": "**:point_right: [Kup teraz]({product_link})**",
"price": "Cena",
"time": "Czas",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Stare SKU**: `{old_sku}`\n**Nowe SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Możliwy nadchodzący drop!"
},
"pt": {
"in_stock_title": "🚀 {gpu_name} EM STOCK!",
"out_of_stock_title": "❌ {gpu_name} esgotado",
"sku_change_title": "🔄 Alteração de SKU detectada para {gpu_name}",
"buy_now": "**:point_right: [Comprar agora]({product_link})**",
"price": "Preço",
"time": "Hora",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**SKU antigo**: `{old_sku}`\n**SKU novo**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Possível drop iminente!"
},
"ro": {
"in_stock_title": "🚀 {gpu_name} ÎN STOC!",
"out_of_stock_title": "❌ {gpu_name} este epuizat",
"sku_change_title": "🔄 Schimbare SKU detectată pentru {gpu_name}",
"buy_now": "**:point_right: [Cumpără acum]({product_link})**",
"price": "Preț",
"time": "Timp",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**SKU vechi**: `{old_sku}`\n**SKU nou**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Posibil drop iminent!"
},
"sk": {
"in_stock_title": "🚀 {gpu_name} NA SKLADE!",
"out_of_stock_title": "❌ {gpu_name} je vypredaný",
"sku_change_title": "🔄 Zmena SKU zistená pre {gpu_name}",
"buy_now": "**:point_right: [Kúp teraz]({product_link})**",
"price": "Cena",
"time": "Čas",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Starý SKU**: `{old_sku}`\n**Nový SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Možný nadchádzajúci drop!"
},
"sl": {
"in_stock_title": "🚀 {gpu_name} NA ZALOGI!",
"out_of_stock_title": "❌ {gpu_name} je razprodan",
"sku_change_title": "🔄 Sprememba SKU zaznana za {gpu_name}",
"buy_now": "**:point_right: [Kupi zdaj]({product_link})**",
"price": "Cena",
"time": "Čas",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Stari SKU**: `{old_sku}`\n**Novi SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Možen prihajajoči drop!"
},
"sv": {
"in_stock_title": "🚀 {gpu_name} I LAGER!",
"out_of_stock_title": "❌ {gpu_name} är slut",
"sku_change_title": "🔄 SKU-ändring upptäckt för {gpu_name}",
"buy_now": "**:point_right: [Köp nu]({product_link})**",
"price": "Pris",
"time": "Tid",
"footer": "NviBot • {DISCORD_SERVER_NAME}",
"sku_description": "**Gammalt SKU**: `{old_sku}`\n**Nytt SKU**: `{new_sku}`",
"imminent_drop": "{DISCORD_ROLE} ⚠️ Möjlig förestående drop!"
}
}

411
app/nvidia-stock-bot.py Normal file
View File

@ -0,0 +1,411 @@
import requests
import logging
import time
import os
import re
import json
from requests.adapters import HTTPAdapter, Retry
# Logger configuration
logging.basicConfig(
level=logging.INFO,
format="%(asctime)s [%(levelname)s] %(message)s",
)
logging.info("Script started")
# Retrieve environment variables
try:
DISCORD_WEBHOOK_URL = os.environ.get('DISCORD_WEBHOOK_URL')
DISCORD_SERVER_NAME = os.environ.get('DISCORD_SERVER_NAME', 'Shared for free')
API_URL_SKU = os.environ.get('API_URL_SKU', 'https://api.nvidia.partners/edge/product/search?page=1&limit=100&locale=fr-fr&Manufacturer=Nvidia')
API_URL_STOCK = os.environ.get('API_URL_STOCK', 'https://api.store.nvidia.com/partner/v1/feinventory?locale=fr-fr&skus=')
REFRESH_TIME = int(os.environ.get('REFRESH_TIME'))
TEST_MODE = os.environ.get('TEST_MODE', 'False').lower() == 'true'
PRODUCT_URL = os.environ.get('PRODUCT_URL', 'https://marketplace.nvidia.com/fr-fr/consumer/graphics-cards/?locale=fr-fr&page=1&limit=12&manufacturer=NVIDIA')
DISCORD_ROLES = os.environ.get('DISCORD_ROLES')
PRODUCT_NAMES = os.environ.get('PRODUCT_NAMES')
if not PRODUCT_NAMES:
logging.error("❌ PRODUCT_NAMES is required but not defined.")
exit(1)
PRODUCT_NAMES = [name.strip() for name in PRODUCT_NAMES.split(',')]
DISCORD_ROLE_MAP = {}
if not DISCORD_ROLES or not DISCORD_ROLES.strip():
logging.warning("⚠️ DISCORD_ROLES not defined or empty. Defaulting all roles to @everyone.")
for name in PRODUCT_NAMES:
DISCORD_ROLE_MAP[name] = '@everyone'
else:
roles = [r.strip() if r.strip() else '@everyone' for r in DISCORD_ROLES.split(',')]
if len(roles) != len(PRODUCT_NAMES):
logging.error("❌ The number of DISCORD_ROLES must match PRODUCT_NAMES.")
exit(1)
for name, role in zip(PRODUCT_NAMES, roles):
if role != '@everyone' and not re.match(r'^<@&\d{17,20}>$', role):
logging.error(f"❌ Invalid DISCORD_ROLE format for {name}: {role}")
exit(1)
DISCORD_ROLE_MAP[name] = role
if not DISCORD_WEBHOOK_URL:
logging.error("❌ DISCORD_WEBHOOK_URL is required but not defined.")
exit(1)
# Regex to extract ID and token
match = re.search(r'/(\d+)/(.*)', DISCORD_WEBHOOK_URL)
if match:
webhook_id = match.group(1)
webhook_token = match.group(2)
# Mask last characters of the ID
masked_webhook_id = webhook_id[:len(webhook_id) - 10] + '*' * 10
# Mask last characters of the token
masked_webhook_token = webhook_token[:len(webhook_token) - 120] + '*' * 10
# Rebuild masked URL
wh_masked_url = f"https://discord.com/api/webhooks/{masked_webhook_id}/{masked_webhook_token}"
# Error logging
except KeyError as e:
logging.error(f"Missing environment variable: {e}")
exit(1)
except ValueError:
logging.error("REFRESH_TIME must be a valid integer.")
exit(1)
# Localization
try:
with open("localization.json", "r", encoding="utf-8") as f:
localization = json.load(f)
except FileNotFoundError:
logging.error("❌ localization.json file not found.")
exit(1)
language = os.environ.get("DISCORD_NOTIFICATION_LANGUAGE", "en").lower()
required_keys = [
"in_stock_title", "out_of_stock_title", "sku_change_title",
"buy_now", "price", "time", "footer",
"sku_description", "imminent_drop"
]
loc = localization.get(language, localization["en"])
logging.info(f"Notification language: {language}")
if not loc:
logging.warning(f"⚠️ Language '{language}' not found. Falling back to English.")
loc = localization.get("en")
language = "en"
if not loc:
logging.error("❌ No localization found for language 'en'. Cannot continue.")
sys.exit(1)
fallback = localization.get("en", {})
missing_keys = [key for key in required_keys if key not in loc]
if missing_keys:
logging.warning(f"⚠️ Missing keys in localization for '{language}': {', '.join(missing_keys)}. Falling back to English for those.")
for key in missing_keys:
if key in fallback:
loc[key] = fallback[key]
else:
logging.error(f"❌ Missing required key '{key}' in both '{language}' and fallback 'en'.")
sys.exit(1)
in_stock_title = loc["in_stock_title"]
out_of_stock_title = loc["out_of_stock_title"]
sku_change_title = loc["sku_change_title"]
buy_now = loc["buy_now"]
price_label = loc["price"]
time_label = loc["time"]
footer = loc["footer"]
sku_description = loc["sku_description"]
imminent_drop = loc["imminent_drop"]
# Display URLs and configurations
logging.info(f"GPU: {PRODUCT_NAMES}")
logging.info(f"Discord Webhook URL: {wh_masked_url}")
logging.info(f"Discord Role Mention: {DISCORD_ROLES}")
logging.info(f"API URL SKU: {API_URL_SKU}")
logging.info(f"API URL Stock: {API_URL_STOCK}")
logging.info(f"Product URL: {PRODUCT_URL}")
logging.info(f"Refresh time: {REFRESH_TIME} seconds")
logging.info(f"Test Mode: {TEST_MODE}")
# HTTP headers
HEADERS = {
"User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) "
"AppleWebKit/537.36 (KHTML, like Gecko) "
"Chrome/131.0.0.0 Safari/537.36",
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,"
"image/avif,image/webp,image/apng,*/*;q=0.8",
"Accept-Encoding": "gzip, deflate, br, zstd",
"Accept-Language": "en-US,en;q=0.5",
"Cache-Control": "max-age=0",
"Connection": "keep-alive",
"Sec-Fetch-Dest": "document",
"Sec-Fetch-Mode": "navigate",
"Sec-Fetch-Site": "none",
"Sec-Fetch-User": "?1",
"Upgrade-Insecure-Requests": "1",
"Sec-GPC": "1",
}
# Session with retries
session = requests.Session()
retries = Retry(total=5, backoff_factor=1, status_forcelist=[500, 502, 503, 504])
session.mount('https://', HTTPAdapter(max_retries=retries))
last_sku_dict = {}
global_stock_status_dict = {}
first_run_dict = {name: True for name in PRODUCT_NAMES}
# Discord notifications
def send_discord_notification(gpu_name: str, product_link: str, products_price: str):
# Get current UNIX timestamp
timestamp_unix = int(time.time())
if TEST_MODE:
logging.info(f"[TEST MODE] Discord Notification: {gpu_name} available!")
return
embed = {
"title": in_stock_title.format(gpu_name=gpu_name),
"color": 3066993,
"thumbnail": {
"url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/RTX5000.jpg"
},
"author": {
"name": "Nvidia Founder Editions"
},
"fields": [
{
"name": price_label,
"value": f"`{products_price}€`",
"inline": True
},
{
"name": time_label,
"value": f"<t:{timestamp_unix}:d> <t:{timestamp_unix}:T>",
"inline": True
},
],
"description": buy_now.format(product_link=product_link),
"footer": {
"text": footer.format(DISCORD_SERVER_NAME=DISCORD_SERVER_NAME),
"icon_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg"
}
}
payload = {
"content": f"{DISCORD_ROLE_MAP.get(gpu_name, '@everyone')}",
"username": "NviBot",
"avatar_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg",
"embeds": [embed]
}
try:
response = requests.post(DISCORD_WEBHOOK_URL, json=payload)
if response.status_code == 204:
logging.info("✅ Notification sent to Discord.")
else:
logging.error(f"❌ Webhook error: {response.status_code} - {response.text}")
except Exception as e:
logging.error(f"🚨 Error sending webhook: {e}")
def send_out_of_stock_notification(gpu_name: str, product_link: str, products_price: str):
# Get current UNIX timestamp
timestamp_unix = int(time.time())
if TEST_MODE:
logging.info(f"[TEST MODE] Discord Notification: {gpu_name} out of stock!")
return
embed = {
"title": out_of_stock_title.format(gpu_name=gpu_name),
"color": 15158332, # Red for out of stock
"thumbnail": {
"url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/RTX5000.jpg"
},
"url": f"{product_link}",
"author": {
"name": "Nvidia Founder Editions"
},
"footer": {
"text": footer.format(DISCORD_SERVER_NAME=DISCORD_SERVER_NAME),
"icon_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg"
},
"fields": [
{
"name": time_label,
"value": f"<t:{timestamp_unix}:d> <t:{timestamp_unix}:T>",
"inline": True
}
]
}
payload = {"username": "NviBot", "avatar_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg", "embeds": [embed]}
try:
response = requests.post(DISCORD_WEBHOOK_URL, json=payload)
if response.status_code == 204:
logging.info("'Out of stock' notification sent to Discord.")
else:
logging.error(f"❌ Webhook error: {response.status_code} - {response.text}")
except Exception as e:
logging.error(f"🚨 Error sending webhook: {e}")
def send_sku_change_notification(gpu_name: str, old_sku: str, new_sku: str, product_link: str):
# Get current UNIX timestamp
timestamp_unix = int(time.time())
if TEST_MODE:
logging.info(f"[TEST MODE] SKU change detected: {old_sku}{new_sku}")
return
embed = {
"title": sku_change_title.format(gpu_name=gpu_name),
"url": f"{product_link}",
"description": sku_description.format(old_sku=old_sku, new_sku=new_sku),
"color": 16776960, # Yellow
"footer": {
"text": footer.format(DISCORD_SERVER_NAME=DISCORD_SERVER_NAME),
"icon_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg"
},
"fields": [
{
"name": time_label,
"value": f"<t:{timestamp_unix}:d> <t:{timestamp_unix}:T>",
"inline": True
}
]
}
payload = {
"content": imminent_drop.format(DISCORD_ROLE=DISCORD_ROLE_MAP.get(gpu_name, '@everyone')),
"username": "NviBot",
"avatar_url": "https://git.djeex.fr/Djeex/nvidia-stock-bot/raw/branch/main/assets/img/ds_wh_pp.jpg",
"embeds": [embed]
}
try:
response = requests.post(DISCORD_WEBHOOK_URL, json=payload)
if response.status_code == 204:
logging.info("✅ SKU change notification sent to Discord.")
else:
logging.error(f"❌ Webhook error: {response.status_code} - {response.text}")
except Exception as e:
logging.error(f"🚨 Error sending webhook: {e}")
# Stock search
def check_rtx_50_founders():
global last_sku_dict, global_stock_status_dict, first_run_dict
try:
response = session.get(API_URL_SKU, headers=HEADERS, timeout=10)
logging.info(f"SKU API response: {response.status_code}")
response.raise_for_status()
data = response.json()
except requests.exceptions.RequestException as e:
logging.error(f"SKU API error: {e}")
return
# All available products
all_products = data['searchedProducts']['productDetails']
for product_name in PRODUCT_NAMES:
product_details = None
for p in all_products:
if p.get("gpu", "").strip() == product_name:
product_details = p
break
if not product_details:
logging.warning(f"⚠️ No product with GPU '{product_name}' found.")
continue
product_sku = product_details['productSKU']
product_upc = product_details.get('productUPC', "")
if not isinstance(product_upc, list):
product_upc = [product_upc]
# Check SKU change
old_sku = last_sku_dict.get(product_name)
if old_sku and old_sku != product_sku and not first_run_dict[product_name]:
logging.warning(f"⚠️ SKU changed for {product_name}: {old_sku}{product_sku}")
send_sku_change_notification(product_name, old_sku, product_sku, PRODUCT_URL)
last_sku_dict[product_name] = product_sku
first_run_dict[product_name] = False
# Stock check
api_stock_url = API_URL_STOCK + product_sku
logging.info(f"[{product_name}] Checking stock: {api_stock_url}")
try:
response = session.get(api_stock_url, headers=HEADERS, timeout=10)
logging.info(f"[{product_name}] Stock API response: {response.status_code}")
response.raise_for_status()
stock_data = response.json()
except requests.exceptions.RequestException as e:
logging.error(f"Stock API error: {e}")
continue
products = stock_data.get("listMap", [])
products_price = "Price not available"
if isinstance(products, list) and len(products) > 0:
for p in products:
price = p.get("price", 'Price not available')
if price != 'Price not available':
products_price = price
break
else:
logging.error(f"[{product_name}] Product list is empty or malformed.")
found_in_stock = set()
for p in products:
gpu_name = p.get("fe_sku", "").upper()
is_active = p.get("is_active") == "true"
if is_active and any(upc.upper() in gpu_name for upc in product_upc):
found_in_stock.add(gpu_name)
for upc in product_upc:
upc_upper = upc.upper()
currently_in_stock = upc_upper in found_in_stock
previously_in_stock = global_stock_status_dict.get((product_name, upc_upper), False)
if currently_in_stock and not previously_in_stock:
send_discord_notification(product_name, PRODUCT_URL, products_price)
global_stock_status_dict[(product_name, upc_upper)] = True
logging.info(f"[{product_name}] {upc} is now in stock!")
elif not currently_in_stock and previously_in_stock:
send_out_of_stock_notification(product_name, PRODUCT_URL, products_price)
global_stock_status_dict[(product_name, upc_upper)] = False
logging.info(f"[{product_name}] {upc} is now out of stock.")
elif currently_in_stock:
logging.info(f"[{product_name}] {upc} still in stock.")
else:
logging.info(f"[{product_name}] {upc} still out of stock.")
# Loop
if __name__ == "__main__":
try:
while True:
start = time.time()
check_rtx_50_founders()
elapsed = time.time() - start
time.sleep(max(0, REFRESH_TIME - elapsed))
# Gracefully shut down
except KeyboardInterrupt:
logging.info("🛑 Script interrupted by user. Exiting gracefully.")
exit(0)

BIN
assets/img/RTX5000.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 19 KiB

BIN
assets/img/ds_wh_pp.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 6.2 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 322 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 95 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 25 KiB

View File

@ -1,4 +1,5 @@
DS_HOOK="votre url du webhook discord"
FREQ=60 #frequence de rafraichissement en secondes
GPU=
URL=""
PRODUCT_NAMES= # Exact GPU name (e.g. "RTX 5080, RTX 5090")
DISCORD_WEBHOOK_URL= # Your Discord webhook URL
API_URL_SKU= # API listing the product for your country
API_URL_STOCK= # API providing stock data for your country
PRODUCT_URL= # GPU purchase URL for your country

View File

@ -1,15 +1,16 @@
version: "3.8"
services:
nvidia-stock-bot:
image: nvidia-stock-bot
container_name: nvidia-stock-bot
restart: always # Le conteneur redémarrera automatiquement en cas d'échec
restart: unless-stopped
env_file:
- .env
environment:
- DISCORD_WEBHOOK_URL=${DS_HOOK}
- REFRESH_TIME=${FREQ}
- GPU_TARGETS=${GPU} #SKU
- API_URL=${URL} #URL de l'API
- PYTHONUNBUFFERED=1 # Permet d'afficher les logs en temps réel
command: python nvidia-stock-bot.py # Lance le script Python
# Minimal environment variables
- PRODUCT_NAMES=${PRODUCT_NAMES} # Exact GPU name (e.g. "RTX 5080, RTX 5090")
- DISCORD_WEBHOOK_URL=${DISCORD_WEBHOOK_URL} # Your Discord webhook URL
- API_URL_SKU=${API_URL_SKU} # API listing the product for your country
- API_URL_STOCK=${API_URL_STOCK} # API providing stock data for your country
- PRODUCT_URL=${PRODUCT_URL} # GPU purchase URL for your country
- PYTHONUNBUFFERED=1 # Enables real-time log output
command: python nvidia-stock-bot.py # Run the script

View File

@ -1,140 +0,0 @@
import requests
import logging
import time
import os
# Configuration du logger
logging.basicConfig(
level=logging.INFO,
format="%(asctime)s [%(levelname)s] %(message)s",
)
logging.info("Démarrage du script")
# Récupération des variables d'environnement
try:
DISCORD_WEBHOOK_URL = os.environ['DISCORD_WEBHOOK_URL']
API_URL = os.environ['API_URL']
GPU_TARGETS = os.environ['GPU_TARGETS'].split(",") # Séparer en liste
GPU_TARGETS = [gpu.strip() for gpu in GPU_TARGETS] # Nettoyer les espaces
REFRESH_TIME = int(os.environ['REFRESH_TIME']) # Convertir en entier
except KeyError as e:
logging.error(f"Variable d'environnement manquante : {e}")
exit(1) # Quitter le script proprement en cas d'erreur
except ValueError:
logging.error("REFRESH_TIME doit être un entier valide.")
exit(1)
# Afficher les valeurs des variables d'environnement
print(f"url du webhook Discord: {DISCORD_WEBHOOK_URL}")
print(f"url de l'API: {API_URL}")
print(f"GPU recherché: {GPU_TARGETS}")
print(f"Temps d'actualisation (en secondes) : {REFRESH_TIME}")
# LURL de lAPI (exemple)
#API_URL = "https://api.store.nvidia.com/partner/v1/feinventory?locale=fr-fr&skus=5090LDLCFE"
# GPUs à surveiller
#GPU_TARGETS = ["5090LDLCFE_FR"]
# Entêtes HTTP pour la requête
HEADERS = {
"User-Agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) "
"AppleWebKit/537.36 (KHTML, like Gecko) "
"Chrome/131.0.0.0 Safari/537.36",
"Accept": "text/html,application/xhtml+xml,application/xml;q=0.9,"
"image/avif,image/webp,image/apng,*/*;q=0.8",
"Accept-Encoding": "gzip, deflate, br, zstd",
"Accept-Language": "en-US,en;q=0.5",
"Cache-Control": "max-age=0",
"Connection": "keep-alive",
"Sec-Fetch-Dest": "document",
"Sec-Fetch-Mode": "navigate",
"Sec-Fetch-Site": "none",
"Sec-Fetch-User": "?1",
"Upgrade-Insecure-Requests": "1",
"Sec-GPC": "1",
}
# Dictionnaire stockant l'état de stock
stock_status = {gpu.upper(): False for gpu in GPU_TARGETS}
session = requests.Session()
def send_discord_notification(gpu_name: str, product_link: str):
"""Envoie une notification Discord avec un embed via un webhook."""
embed = {
"title": f"🚀 {gpu_name} en stock !",
"description": f":point_right: **[Achetez ici](https://marketplace.nvidia.com/fr-fr/consumer/graphics-cards/?locale=fr-fr&page=1&limit=12&gpu=RTX%205090,RTX%205080&manufacturer=NVIDIA)**",
"color": 3066993, # Couleur verte
"timestamp": time.strftime("%Y-%m-%dT%H:%M:%S", time.gmtime()),
#"thumbnail": {
# "url": "https://www.nvidia.com/content/dam/en-zz/Solutions/geforce/graphic-cards/50-series/rtx-5090/geforce-rtx-5090-learn-more-og-1200x630.jpg"
#}
}
payload = {
"content": "@everyone",
"username": "Nvidia Bot",
"embeds": [embed]
}
try:
response = requests.post(DISCORD_WEBHOOK_URL, json=payload)
if response.status_code == 204:
logging.info("✅ Embed envoyé sur Discord.")
else:
logging.error(f"❌ Erreur d'envoi du webhook : {response.status_code} - {response.text}")
except Exception as e:
logging.error(f"🚨 Erreur lors de l'envoi du webhook : {e}")
def check_rtx_50_founders():
"""Vérifie l'état de stock des GPU Founders Edition et notifie Discord si un GPU repasse en stock."""
try:
response = session.get(API_URL, headers=HEADERS, timeout=10)
logging.info(f"Réponse de l'API : {response.status_code}")
response.raise_for_status()
data = response.json()
except requests.exceptions.RequestException as e:
logging.error(f"Erreur lors de l'appel API : {e}")
return
products = data.get("listMap", [])
found_in_stock = set()
for p in products:
gpu_name = p.get("fe_sku", "").upper()
is_active = p.get("is_active") == "true"
if is_active:
if any(target.upper() in gpu_name for target in GPU_TARGETS):
found_in_stock.add(gpu_name)
for gpu in GPU_TARGETS:
gpu_upper = gpu.upper()
currently_in_stock = (gpu_upper in found_in_stock)
previously_in_stock = stock_status[gpu_upper]
if currently_in_stock and not previously_in_stock:
for product in products:
product_name = product.get("fe_sku", "").upper()
if product_name == gpu_upper:
real_gpu_name = product.get("fe_sku", "Inconnu")
product_link = "https://marketplace.nvidia.com/fr-fr/consumer/graphics-cards/?locale=fr-fr&page=1&limit=12&gpu=RTX%205090,RTX%205080"
send_discord_notification(real_gpu_name, product_link)
stock_status[gpu_upper] = True
print(f"{gpu} est maintenant en stock!")
elif (not currently_in_stock) and previously_in_stock:
logging.info(f"{gpu} n'est plus en stock.")
stock_status[gpu_upper] = False
print(f"{gpu} est hors stock !")
elif not currently_in_stock:
print(f"{gpu} est actuellement hors stock.")
if __name__ == "__main__":
while True:
check_rtx_50_founders()
time.sleep(REFRESH_TIME)