samuelrince
commited on
Commit
•
85e37e1
1
Parent(s):
56b18b9
feat: massive update :)
Browse files- LICENSE +426 -201
- README.md +9 -7
- app.py +323 -298
- src/assets.py +6 -0
- src/constants.py +82 -0
- src/content.py +229 -0
- {data → src}/electricity_mix.csv +0 -0
- data/mixes.py → src/electricity_mix.py +4 -3
- src/utils.py +182 -0
LICENSE
CHANGED
@@ -1,202 +1,427 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
|
2 |
-
|
3 |
-
|
4 |
-
|
5 |
-
|
6 |
-
|
7 |
-
|
8 |
-
|
9 |
-
|
10 |
-
|
11 |
-
|
12 |
-
|
13 |
-
|
14 |
-
|
15 |
-
|
16 |
-
|
17 |
-
|
18 |
-
|
19 |
-
|
20 |
-
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
|
50 |
-
|
51 |
-
|
52 |
-
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
|
62 |
-
|
63 |
-
|
64 |
-
|
65 |
-
|
66 |
-
|
67 |
-
|
68 |
-
|
69 |
-
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
85 |
-
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
|
93 |
-
|
94 |
-
|
95 |
-
|
96 |
-
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
|
114 |
-
|
115 |
-
|
116 |
-
|
117 |
-
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
136 |
-
|
137 |
-
|
138 |
-
|
139 |
-
|
140 |
-
|
141 |
-
|
142 |
-
|
143 |
-
|
144 |
-
|
145 |
-
|
146 |
-
|
147 |
-
|
148 |
-
|
149 |
-
|
150 |
-
|
151 |
-
|
152 |
-
|
153 |
-
|
154 |
-
|
155 |
-
|
156 |
-
|
157 |
-
|
158 |
-
|
159 |
-
|
160 |
-
|
161 |
-
|
162 |
-
|
163 |
-
|
164 |
-
|
165 |
-
|
166 |
-
|
167 |
-
|
168 |
-
|
169 |
-
|
170 |
-
|
171 |
-
|
172 |
-
|
173 |
-
|
174 |
-
|
175 |
-
|
176 |
-
|
177 |
-
|
178 |
-
|
179 |
-
|
180 |
-
|
181 |
-
|
182 |
-
|
183 |
-
|
184 |
-
|
185 |
-
|
186 |
-
|
187 |
-
|
188 |
-
|
189 |
-
|
190 |
-
|
191 |
-
|
192 |
-
|
193 |
-
|
194 |
-
|
195 |
-
|
196 |
-
|
197 |
-
|
198 |
-
|
199 |
-
|
200 |
-
|
201 |
-
|
202 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
Attribution-ShareAlike 4.0 International
|
2 |
+
|
3 |
+
=======================================================================
|
4 |
+
|
5 |
+
Creative Commons Corporation ("Creative Commons") is not a law firm and
|
6 |
+
does not provide legal services or legal advice. Distribution of
|
7 |
+
Creative Commons public licenses does not create a lawyer-client or
|
8 |
+
other relationship. Creative Commons makes its licenses and related
|
9 |
+
information available on an "as-is" basis. Creative Commons gives no
|
10 |
+
warranties regarding its licenses, any material licensed under their
|
11 |
+
terms and conditions, or any related information. Creative Commons
|
12 |
+
disclaims all liability for damages resulting from their use to the
|
13 |
+
fullest extent possible.
|
14 |
+
|
15 |
+
Using Creative Commons Public Licenses
|
16 |
+
|
17 |
+
Creative Commons public licenses provide a standard set of terms and
|
18 |
+
conditions that creators and other rights holders may use to share
|
19 |
+
original works of authorship and other material subject to copyright
|
20 |
+
and certain other rights specified in the public license below. The
|
21 |
+
following considerations are for informational purposes only, are not
|
22 |
+
exhaustive, and do not form part of our licenses.
|
23 |
+
|
24 |
+
Considerations for licensors: Our public licenses are
|
25 |
+
intended for use by those authorized to give the public
|
26 |
+
permission to use material in ways otherwise restricted by
|
27 |
+
copyright and certain other rights. Our licenses are
|
28 |
+
irrevocable. Licensors should read and understand the terms
|
29 |
+
and conditions of the license they choose before applying it.
|
30 |
+
Licensors should also secure all rights necessary before
|
31 |
+
applying our licenses so that the public can reuse the
|
32 |
+
material as expected. Licensors should clearly mark any
|
33 |
+
material not subject to the license. This includes other CC-
|
34 |
+
licensed material, or material used under an exception or
|
35 |
+
limitation to copyright. More considerations for licensors:
|
36 |
+
wiki.creativecommons.org/Considerations_for_licensors
|
37 |
+
|
38 |
+
Considerations for the public: By using one of our public
|
39 |
+
licenses, a licensor grants the public permission to use the
|
40 |
+
licensed material under specified terms and conditions. If
|
41 |
+
the licensor's permission is not necessary for any reason--for
|
42 |
+
example, because of any applicable exception or limitation to
|
43 |
+
copyright--then that use is not regulated by the license. Our
|
44 |
+
licenses grant only permissions under copyright and certain
|
45 |
+
other rights that a licensor has authority to grant. Use of
|
46 |
+
the licensed material may still be restricted for other
|
47 |
+
reasons, including because others have copyright or other
|
48 |
+
rights in the material. A licensor may make special requests,
|
49 |
+
such as asking that all changes be marked or described.
|
50 |
+
Although not required by our licenses, you are encouraged to
|
51 |
+
respect those requests where reasonable. More considerations
|
52 |
+
for the public:
|
53 |
+
wiki.creativecommons.org/Considerations_for_licensees
|
54 |
+
|
55 |
+
=======================================================================
|
56 |
+
|
57 |
+
Creative Commons Attribution-ShareAlike 4.0 International Public
|
58 |
+
License
|
59 |
+
|
60 |
+
By exercising the Licensed Rights (defined below), You accept and agree
|
61 |
+
to be bound by the terms and conditions of this Creative Commons
|
62 |
+
Attribution-ShareAlike 4.0 International Public License ("Public
|
63 |
+
License"). To the extent this Public License may be interpreted as a
|
64 |
+
contract, You are granted the Licensed Rights in consideration of Your
|
65 |
+
acceptance of these terms and conditions, and the Licensor grants You
|
66 |
+
such rights in consideration of benefits the Licensor receives from
|
67 |
+
making the Licensed Material available under these terms and
|
68 |
+
conditions.
|
69 |
+
|
70 |
+
|
71 |
+
Section 1 -- Definitions.
|
72 |
+
|
73 |
+
a. Adapted Material means material subject to Copyright and Similar
|
74 |
+
Rights that is derived from or based upon the Licensed Material
|
75 |
+
and in which the Licensed Material is translated, altered,
|
76 |
+
arranged, transformed, or otherwise modified in a manner requiring
|
77 |
+
permission under the Copyright and Similar Rights held by the
|
78 |
+
Licensor. For purposes of this Public License, where the Licensed
|
79 |
+
Material is a musical work, performance, or sound recording,
|
80 |
+
Adapted Material is always produced where the Licensed Material is
|
81 |
+
synched in timed relation with a moving image.
|
82 |
+
|
83 |
+
b. Adapter's License means the license You apply to Your Copyright
|
84 |
+
and Similar Rights in Your contributions to Adapted Material in
|
85 |
+
accordance with the terms and conditions of this Public License.
|
86 |
+
|
87 |
+
c. BY-SA Compatible License means a license listed at
|
88 |
+
creativecommons.org/compatiblelicenses, approved by Creative
|
89 |
+
Commons as essentially the equivalent of this Public License.
|
90 |
+
|
91 |
+
d. Copyright and Similar Rights means copyright and/or similar rights
|
92 |
+
closely related to copyright including, without limitation,
|
93 |
+
performance, broadcast, sound recording, and Sui Generis Database
|
94 |
+
Rights, without regard to how the rights are labeled or
|
95 |
+
categorized. For purposes of this Public License, the rights
|
96 |
+
specified in Section 2(b)(1)-(2) are not Copyright and Similar
|
97 |
+
Rights.
|
98 |
+
|
99 |
+
e. Effective Technological Measures means those measures that, in the
|
100 |
+
absence of proper authority, may not be circumvented under laws
|
101 |
+
fulfilling obligations under Article 11 of the WIPO Copyright
|
102 |
+
Treaty adopted on December 20, 1996, and/or similar international
|
103 |
+
agreements.
|
104 |
+
|
105 |
+
f. Exceptions and Limitations means fair use, fair dealing, and/or
|
106 |
+
any other exception or limitation to Copyright and Similar Rights
|
107 |
+
that applies to Your use of the Licensed Material.
|
108 |
+
|
109 |
+
g. License Elements means the license attributes listed in the name
|
110 |
+
of a Creative Commons Public License. The License Elements of this
|
111 |
+
Public License are Attribution and ShareAlike.
|
112 |
+
|
113 |
+
h. Licensed Material means the artistic or literary work, database,
|
114 |
+
or other material to which the Licensor applied this Public
|
115 |
+
License.
|
116 |
+
|
117 |
+
i. Licensed Rights means the rights granted to You subject to the
|
118 |
+
terms and conditions of this Public License, which are limited to
|
119 |
+
all Copyright and Similar Rights that apply to Your use of the
|
120 |
+
Licensed Material and that the Licensor has authority to license.
|
121 |
+
|
122 |
+
j. Licensor means the individual(s) or entity(ies) granting rights
|
123 |
+
under this Public License.
|
124 |
+
|
125 |
+
k. Share means to provide material to the public by any means or
|
126 |
+
process that requires permission under the Licensed Rights, such
|
127 |
+
as reproduction, public display, public performance, distribution,
|
128 |
+
dissemination, communication, or importation, and to make material
|
129 |
+
available to the public including in ways that members of the
|
130 |
+
public may access the material from a place and at a time
|
131 |
+
individually chosen by them.
|
132 |
+
|
133 |
+
l. Sui Generis Database Rights means rights other than copyright
|
134 |
+
resulting from Directive 96/9/EC of the European Parliament and of
|
135 |
+
the Council of 11 March 1996 on the legal protection of databases,
|
136 |
+
as amended and/or succeeded, as well as other essentially
|
137 |
+
equivalent rights anywhere in the world.
|
138 |
+
|
139 |
+
m. You means the individual or entity exercising the Licensed Rights
|
140 |
+
under this Public License. Your has a corresponding meaning.
|
141 |
+
|
142 |
+
|
143 |
+
Section 2 -- Scope.
|
144 |
+
|
145 |
+
a. License grant.
|
146 |
+
|
147 |
+
1. Subject to the terms and conditions of this Public License,
|
148 |
+
the Licensor hereby grants You a worldwide, royalty-free,
|
149 |
+
non-sublicensable, non-exclusive, irrevocable license to
|
150 |
+
exercise the Licensed Rights in the Licensed Material to:
|
151 |
+
|
152 |
+
a. reproduce and Share the Licensed Material, in whole or
|
153 |
+
in part; and
|
154 |
+
|
155 |
+
b. produce, reproduce, and Share Adapted Material.
|
156 |
+
|
157 |
+
2. Exceptions and Limitations. For the avoidance of doubt, where
|
158 |
+
Exceptions and Limitations apply to Your use, this Public
|
159 |
+
License does not apply, and You do not need to comply with
|
160 |
+
its terms and conditions.
|
161 |
+
|
162 |
+
3. Term. The term of this Public License is specified in Section
|
163 |
+
6(a).
|
164 |
+
|
165 |
+
4. Media and formats; technical modifications allowed. The
|
166 |
+
Licensor authorizes You to exercise the Licensed Rights in
|
167 |
+
all media and formats whether now known or hereafter created,
|
168 |
+
and to make technical modifications necessary to do so. The
|
169 |
+
Licensor waives and/or agrees not to assert any right or
|
170 |
+
authority to forbid You from making technical modifications
|
171 |
+
necessary to exercise the Licensed Rights, including
|
172 |
+
technical modifications necessary to circumvent Effective
|
173 |
+
Technological Measures. For purposes of this Public License,
|
174 |
+
simply making modifications authorized by this Section 2(a)
|
175 |
+
(4) never produces Adapted Material.
|
176 |
+
|
177 |
+
5. Downstream recipients.
|
178 |
+
|
179 |
+
a. Offer from the Licensor -- Licensed Material. Every
|
180 |
+
recipient of the Licensed Material automatically
|
181 |
+
receives an offer from the Licensor to exercise the
|
182 |
+
Licensed Rights under the terms and conditions of this
|
183 |
+
Public License.
|
184 |
+
|
185 |
+
b. Additional offer from the Licensor -- Adapted Material.
|
186 |
+
Every recipient of Adapted Material from You
|
187 |
+
automatically receives an offer from the Licensor to
|
188 |
+
exercise the Licensed Rights in the Adapted Material
|
189 |
+
under the conditions of the Adapter's License You apply.
|
190 |
+
|
191 |
+
c. No downstream restrictions. You may not offer or impose
|
192 |
+
any additional or different terms or conditions on, or
|
193 |
+
apply any Effective Technological Measures to, the
|
194 |
+
Licensed Material if doing so restricts exercise of the
|
195 |
+
Licensed Rights by any recipient of the Licensed
|
196 |
+
Material.
|
197 |
|
198 |
+
6. No endorsement. Nothing in this Public License constitutes or
|
199 |
+
may be construed as permission to assert or imply that You
|
200 |
+
are, or that Your use of the Licensed Material is, connected
|
201 |
+
with, or sponsored, endorsed, or granted official status by,
|
202 |
+
the Licensor or others designated to receive attribution as
|
203 |
+
provided in Section 3(a)(1)(A)(i).
|
204 |
+
|
205 |
+
b. Other rights.
|
206 |
+
|
207 |
+
1. Moral rights, such as the right of integrity, are not
|
208 |
+
licensed under this Public License, nor are publicity,
|
209 |
+
privacy, and/or other similar personality rights; however, to
|
210 |
+
the extent possible, the Licensor waives and/or agrees not to
|
211 |
+
assert any such rights held by the Licensor to the limited
|
212 |
+
extent necessary to allow You to exercise the Licensed
|
213 |
+
Rights, but not otherwise.
|
214 |
+
|
215 |
+
2. Patent and trademark rights are not licensed under this
|
216 |
+
Public License.
|
217 |
+
|
218 |
+
3. To the extent possible, the Licensor waives any right to
|
219 |
+
collect royalties from You for the exercise of the Licensed
|
220 |
+
Rights, whether directly or through a collecting society
|
221 |
+
under any voluntary or waivable statutory or compulsory
|
222 |
+
licensing scheme. In all other cases the Licensor expressly
|
223 |
+
reserves any right to collect such royalties.
|
224 |
+
|
225 |
+
|
226 |
+
Section 3 -- License Conditions.
|
227 |
+
|
228 |
+
Your exercise of the Licensed Rights is expressly made subject to the
|
229 |
+
following conditions.
|
230 |
+
|
231 |
+
a. Attribution.
|
232 |
+
|
233 |
+
1. If You Share the Licensed Material (including in modified
|
234 |
+
form), You must:
|
235 |
+
|
236 |
+
a. retain the following if it is supplied by the Licensor
|
237 |
+
with the Licensed Material:
|
238 |
+
|
239 |
+
i. identification of the creator(s) of the Licensed
|
240 |
+
Material and any others designated to receive
|
241 |
+
attribution, in any reasonable manner requested by
|
242 |
+
the Licensor (including by pseudonym if
|
243 |
+
designated);
|
244 |
+
|
245 |
+
ii. a copyright notice;
|
246 |
+
|
247 |
+
iii. a notice that refers to this Public License;
|
248 |
+
|
249 |
+
iv. a notice that refers to the disclaimer of
|
250 |
+
warranties;
|
251 |
+
|
252 |
+
v. a URI or hyperlink to the Licensed Material to the
|
253 |
+
extent reasonably practicable;
|
254 |
+
|
255 |
+
b. indicate if You modified the Licensed Material and
|
256 |
+
retain an indication of any previous modifications; and
|
257 |
+
|
258 |
+
c. indicate the Licensed Material is licensed under this
|
259 |
+
Public License, and include the text of, or the URI or
|
260 |
+
hyperlink to, this Public License.
|
261 |
+
|
262 |
+
2. You may satisfy the conditions in Section 3(a)(1) in any
|
263 |
+
reasonable manner based on the medium, means, and context in
|
264 |
+
which You Share the Licensed Material. For example, it may be
|
265 |
+
reasonable to satisfy the conditions by providing a URI or
|
266 |
+
hyperlink to a resource that includes the required
|
267 |
+
information.
|
268 |
+
|
269 |
+
3. If requested by the Licensor, You must remove any of the
|
270 |
+
information required by Section 3(a)(1)(A) to the extent
|
271 |
+
reasonably practicable.
|
272 |
+
|
273 |
+
b. ShareAlike.
|
274 |
+
|
275 |
+
In addition to the conditions in Section 3(a), if You Share
|
276 |
+
Adapted Material You produce, the following conditions also apply.
|
277 |
+
|
278 |
+
1. The Adapter's License You apply must be a Creative Commons
|
279 |
+
license with the same License Elements, this version or
|
280 |
+
later, or a BY-SA Compatible License.
|
281 |
+
|
282 |
+
2. You must include the text of, or the URI or hyperlink to, the
|
283 |
+
Adapter's License You apply. You may satisfy this condition
|
284 |
+
in any reasonable manner based on the medium, means, and
|
285 |
+
context in which You Share Adapted Material.
|
286 |
+
|
287 |
+
3. You may not offer or impose any additional or different terms
|
288 |
+
or conditions on, or apply any Effective Technological
|
289 |
+
Measures to, Adapted Material that restrict exercise of the
|
290 |
+
rights granted under the Adapter's License You apply.
|
291 |
+
|
292 |
+
|
293 |
+
Section 4 -- Sui Generis Database Rights.
|
294 |
+
|
295 |
+
Where the Licensed Rights include Sui Generis Database Rights that
|
296 |
+
apply to Your use of the Licensed Material:
|
297 |
+
|
298 |
+
a. for the avoidance of doubt, Section 2(a)(1) grants You the right
|
299 |
+
to extract, reuse, reproduce, and Share all or a substantial
|
300 |
+
portion of the contents of the database;
|
301 |
+
|
302 |
+
b. if You include all or a substantial portion of the database
|
303 |
+
contents in a database in which You have Sui Generis Database
|
304 |
+
Rights, then the database in which You have Sui Generis Database
|
305 |
+
Rights (but not its individual contents) is Adapted Material,
|
306 |
+
including for purposes of Section 3(b); and
|
307 |
+
|
308 |
+
c. You must comply with the conditions in Section 3(a) if You Share
|
309 |
+
all or a substantial portion of the contents of the database.
|
310 |
+
|
311 |
+
For the avoidance of doubt, this Section 4 supplements and does not
|
312 |
+
replace Your obligations under this Public License where the Licensed
|
313 |
+
Rights include other Copyright and Similar Rights.
|
314 |
+
|
315 |
+
|
316 |
+
Section 5 -- Disclaimer of Warranties and Limitation of Liability.
|
317 |
+
|
318 |
+
a. UNLESS OTHERWISE SEPARATELY UNDERTAKEN BY THE LICENSOR, TO THE
|
319 |
+
EXTENT POSSIBLE, THE LICENSOR OFFERS THE LICENSED MATERIAL AS-IS
|
320 |
+
AND AS-AVAILABLE, AND MAKES NO REPRESENTATIONS OR WARRANTIES OF
|
321 |
+
ANY KIND CONCERNING THE LICENSED MATERIAL, WHETHER EXPRESS,
|
322 |
+
IMPLIED, STATUTORY, OR OTHER. THIS INCLUDES, WITHOUT LIMITATION,
|
323 |
+
WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
324 |
+
PURPOSE, NON-INFRINGEMENT, ABSENCE OF LATENT OR OTHER DEFECTS,
|
325 |
+
ACCURACY, OR THE PRESENCE OR ABSENCE OF ERRORS, WHETHER OR NOT
|
326 |
+
KNOWN OR DISCOVERABLE. WHERE DISCLAIMERS OF WARRANTIES ARE NOT
|
327 |
+
ALLOWED IN FULL OR IN PART, THIS DISCLAIMER MAY NOT APPLY TO YOU.
|
328 |
+
|
329 |
+
b. TO THE EXTENT POSSIBLE, IN NO EVENT WILL THE LICENSOR BE LIABLE
|
330 |
+
TO YOU ON ANY LEGAL THEORY (INCLUDING, WITHOUT LIMITATION,
|
331 |
+
NEGLIGENCE) OR OTHERWISE FOR ANY DIRECT, SPECIAL, INDIRECT,
|
332 |
+
INCIDENTAL, CONSEQUENTIAL, PUNITIVE, EXEMPLARY, OR OTHER LOSSES,
|
333 |
+
COSTS, EXPENSES, OR DAMAGES ARISING OUT OF THIS PUBLIC LICENSE OR
|
334 |
+
USE OF THE LICENSED MATERIAL, EVEN IF THE LICENSOR HAS BEEN
|
335 |
+
ADVISED OF THE POSSIBILITY OF SUCH LOSSES, COSTS, EXPENSES, OR
|
336 |
+
DAMAGES. WHERE A LIMITATION OF LIABILITY IS NOT ALLOWED IN FULL OR
|
337 |
+
IN PART, THIS LIMITATION MAY NOT APPLY TO YOU.
|
338 |
+
|
339 |
+
c. The disclaimer of warranties and limitation of liability provided
|
340 |
+
above shall be interpreted in a manner that, to the extent
|
341 |
+
possible, most closely approximates an absolute disclaimer and
|
342 |
+
waiver of all liability.
|
343 |
+
|
344 |
+
|
345 |
+
Section 6 -- Term and Termination.
|
346 |
+
|
347 |
+
a. This Public License applies for the term of the Copyright and
|
348 |
+
Similar Rights licensed here. However, if You fail to comply with
|
349 |
+
this Public License, then Your rights under this Public License
|
350 |
+
terminate automatically.
|
351 |
+
|
352 |
+
b. Where Your right to use the Licensed Material has terminated under
|
353 |
+
Section 6(a), it reinstates:
|
354 |
+
|
355 |
+
1. automatically as of the date the violation is cured, provided
|
356 |
+
it is cured within 30 days of Your discovery of the
|
357 |
+
violation; or
|
358 |
+
|
359 |
+
2. upon express reinstatement by the Licensor.
|
360 |
+
|
361 |
+
For the avoidance of doubt, this Section 6(b) does not affect any
|
362 |
+
right the Licensor may have to seek remedies for Your violations
|
363 |
+
of this Public License.
|
364 |
+
|
365 |
+
c. For the avoidance of doubt, the Licensor may also offer the
|
366 |
+
Licensed Material under separate terms or conditions or stop
|
367 |
+
distributing the Licensed Material at any time; however, doing so
|
368 |
+
will not terminate this Public License.
|
369 |
+
|
370 |
+
d. Sections 1, 5, 6, 7, and 8 survive termination of this Public
|
371 |
+
License.
|
372 |
+
|
373 |
+
|
374 |
+
Section 7 -- Other Terms and Conditions.
|
375 |
+
|
376 |
+
a. The Licensor shall not be bound by any additional or different
|
377 |
+
terms or conditions communicated by You unless expressly agreed.
|
378 |
+
|
379 |
+
b. Any arrangements, understandings, or agreements regarding the
|
380 |
+
Licensed Material not stated herein are separate from and
|
381 |
+
independent of the terms and conditions of this Public License.
|
382 |
+
|
383 |
+
|
384 |
+
Section 8 -- Interpretation.
|
385 |
+
|
386 |
+
a. For the avoidance of doubt, this Public License does not, and
|
387 |
+
shall not be interpreted to, reduce, limit, restrict, or impose
|
388 |
+
conditions on any use of the Licensed Material that could lawfully
|
389 |
+
be made without permission under this Public License.
|
390 |
+
|
391 |
+
b. To the extent possible, if any provision of this Public License is
|
392 |
+
deemed unenforceable, it shall be automatically reformed to the
|
393 |
+
minimum extent necessary to make it enforceable. If the provision
|
394 |
+
cannot be reformed, it shall be severed from this Public License
|
395 |
+
without affecting the enforceability of the remaining terms and
|
396 |
+
conditions.
|
397 |
+
|
398 |
+
c. No term or condition of this Public License will be waived and no
|
399 |
+
failure to comply consented to unless expressly agreed to by the
|
400 |
+
Licensor.
|
401 |
+
|
402 |
+
d. Nothing in this Public License constitutes or may be interpreted
|
403 |
+
as a limitation upon, or waiver of, any privileges and immunities
|
404 |
+
that apply to the Licensor or You, including from the legal
|
405 |
+
processes of any jurisdiction or authority.
|
406 |
+
|
407 |
+
|
408 |
+
=======================================================================
|
409 |
+
|
410 |
+
Creative Commons is not a party to its public
|
411 |
+
licenses. Notwithstanding, Creative Commons may elect to apply one of
|
412 |
+
its public licenses to material it publishes and in those instances
|
413 |
+
will be considered the “Licensor.” The text of the Creative Commons
|
414 |
+
public licenses is dedicated to the public domain under the CC0 Public
|
415 |
+
Domain Dedication. Except for the limited purpose of indicating that
|
416 |
+
material is shared under a Creative Commons public license or as
|
417 |
+
otherwise permitted by the Creative Commons policies published at
|
418 |
+
creativecommons.org/policies, Creative Commons does not authorize the
|
419 |
+
use of the trademark "Creative Commons" or any other trademark or logo
|
420 |
+
of Creative Commons without its prior written consent including,
|
421 |
+
without limitation, in connection with any unauthorized modifications
|
422 |
+
to any of its public licenses or any other arrangements,
|
423 |
+
understandings, or agreements concerning use of licensed material. For
|
424 |
+
the avoidance of doubt, this paragraph does not form part of the
|
425 |
+
public licenses.
|
426 |
+
|
427 |
+
Creative Commons may be contacted at creativecommons.org.
|
README.md
CHANGED
@@ -1,13 +1,13 @@
|
|
1 |
---
|
2 |
title: EcoLogits Calculator
|
3 |
-
emoji:
|
4 |
-
colorFrom:
|
5 |
-
colorTo:
|
6 |
sdk: gradio
|
7 |
-
sdk_version: 4.
|
8 |
app_file: app.py
|
9 |
pinned: true
|
10 |
-
license:
|
11 |
---
|
12 |
|
13 |
Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference
|
@@ -33,5 +33,7 @@ To do:
|
|
33 |
- [X] Expose more inputs like the electricity mix
|
34 |
- [X] Examples of electricity mixes
|
35 |
- [X] Custom number of parameters
|
36 |
-
- [
|
37 |
-
- [ ] Idea : "estimate a given prompt impact" function which allows to enter a prompt in a text field and estimate its impacts
|
|
|
|
|
|
1 |
---
|
2 |
title: EcoLogits Calculator
|
3 |
+
emoji: 🧮
|
4 |
+
colorFrom: green
|
5 |
+
colorTo: indigo
|
6 |
sdk: gradio
|
7 |
+
sdk_version: 4.36.1
|
8 |
app_file: app.py
|
9 |
pinned: true
|
10 |
+
license: cc-by-sa-4.0
|
11 |
---
|
12 |
|
13 |
Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference
|
|
|
33 |
- [X] Expose more inputs like the electricity mix
|
34 |
- [X] Examples of electricity mixes
|
35 |
- [X] Custom number of parameters
|
36 |
+
- [x] Live reload mode
|
37 |
+
- [ ] Idea : "estimate a given prompt impact" function which allows to enter a prompt in a text field and estimate its impacts
|
38 |
+
- [ ] Idea: copy-paste a ChatGPT conversation link and estimate the impacts.
|
39 |
+
- [ ] Idea: compare with country electricity consumption
|
app.py
CHANGED
@@ -1,137 +1,85 @@
|
|
|
|
1 |
import gradio as gr
|
2 |
-
from pint import UnitRegistry
|
3 |
|
4 |
-
from data.mixes import MIXES, find_mix
|
5 |
|
6 |
from ecologits.tracers.utils import compute_llm_impacts, _avg
|
7 |
from ecologits.impacts.llm import compute_llm_impacts as compute_llm_impacts_expert
|
8 |
from ecologits.impacts.llm import IF_ELECTRICITY_MIX_GWP, IF_ELECTRICITY_MIX_ADPE, IF_ELECTRICITY_MIX_PE
|
9 |
from ecologits.model_repository import models
|
10 |
|
11 |
-
|
12 |
-
|
13 |
-
|
14 |
-
|
15 |
-
|
16 |
-
|
17 |
-
|
18 |
-
|
19 |
-
|
20 |
-
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
("Mistral AI / Small", "mistralai/mistral-small-2402"),
|
39 |
-
("Mistral AI / Medium", "mistralai/mistral-medium-2312"),
|
40 |
-
("Mistral AI / Large", "mistralai/mistral-large-2402"),
|
41 |
-
("Meta / Llama 3 8B", "huggingface_hub/meta-llama/Meta-Llama-3-8B"),
|
42 |
-
("Meta / Llama 3 70B", "huggingface_hub/meta-llama/Meta-Llama-3-70B"),
|
43 |
-
("Meta / Llama 2 7B", "huggingface_hub/meta-llama/Llama-2-7b-hf"),
|
44 |
-
("Meta / Llama 2 13B", "huggingface_hub/meta-llama/Llama-2-13b-hf"),
|
45 |
-
("Meta / Llama 2 70B", "huggingface_hub/meta-llama/Llama-2-70b-hf"),
|
46 |
-
("Cohere / Command Light", "cohere/command-light"),
|
47 |
-
("Cohere / Command", "cohere/command"),
|
48 |
-
("Cohere / Command R", "cohere/command-r"),
|
49 |
-
("Cohere / Command R+", "cohere/command-r-plus"),
|
50 |
-
]
|
51 |
-
|
52 |
-
PROMPTS = [
|
53 |
-
("Write a Tweet", 50),
|
54 |
-
("Write an email", 170),
|
55 |
-
("Write an article summary", 250),
|
56 |
-
("Small conversation with a chatbot", 400),
|
57 |
-
("Write a report of 5 pages", 5000),
|
58 |
-
]
|
59 |
-
PROMPTS = [(s + f" ({v} output tokens)", v) for (s, v) in PROMPTS]
|
60 |
-
|
61 |
-
|
62 |
-
def format_indicator(name: str, value: str, unit: str) -> str:
|
63 |
-
return f"""
|
64 |
-
## {name}
|
65 |
-
$$ \LARGE {value} \ \large {unit} $$
|
66 |
-
"""
|
67 |
-
|
68 |
-
|
69 |
-
def form_output(impacts):
|
70 |
-
energy_ = q(impacts.energy.value, impacts.energy.unit)
|
71 |
-
eq_energy_ = q(impacts.energy.value * 2, 'km')
|
72 |
-
if energy_ < q("1 kWh"):
|
73 |
-
energy_ = energy_.to("Wh")
|
74 |
-
eq_energy_ = q(impacts.energy.value * 2000, 'm')
|
75 |
-
|
76 |
-
gwp_ = q(impacts.gwp.value, impacts.gwp.unit)
|
77 |
-
eq_gwp_ = q(impacts.gwp.value / 0.032, 'episodes')
|
78 |
-
if gwp_ < q("1 kgCO2eq"):
|
79 |
-
gwp_ = gwp_.to("1 gCO2eq")
|
80 |
-
eq_gwp_ = q(impacts.gwp.value / 0.032, 'episodes')
|
81 |
-
adpe_ = q(impacts.adpe.value, impacts.adpe.unit)
|
82 |
-
|
83 |
-
pe_ = q(impacts.pe.value, impacts.pe.unit)
|
84 |
-
if pe_ < q("1 MJ"):
|
85 |
-
pe_ = pe_.to("kJ")
|
86 |
-
|
87 |
-
return (
|
88 |
-
format_indicator("⚡️ Energy", f"{energy_.magnitude:.3g}", energy_.units),
|
89 |
-
format_indicator("🌍 GHG Emissions", f"{gwp_.magnitude:.3g}", gwp_.units),
|
90 |
-
format_indicator("🪨 Abiotic Resources", f"{adpe_.magnitude:.3g}", adpe_.units),
|
91 |
-
format_indicator("⛽️ Primary Energy", f"{pe_.magnitude:.3g}", pe_.units),
|
92 |
-
format_indicator("🔋 Equivalent energy: distance with a small electric car", f"{eq_energy_.magnitude:.3g}", eq_energy_.units),
|
93 |
-
format_indicator("🏰 Equivalent emissions for 1000 prompts: watching GoT in streaming", f"{eq_gwp_.magnitude:.3g}", eq_gwp_.units)
|
94 |
-
)
|
95 |
-
|
96 |
-
|
97 |
-
def form(
|
98 |
-
model_name: str,
|
99 |
-
prompt_generated_tokens: int
|
100 |
-
):
|
101 |
-
provider, model_name = model_name.split('/', 1)
|
102 |
-
impacts = compute_llm_impacts(
|
103 |
-
provider=provider,
|
104 |
-
model_name=model_name,
|
105 |
-
output_token_count=prompt_generated_tokens,
|
106 |
-
request_latency=100000
|
107 |
-
)
|
108 |
-
return form_output(impacts)
|
109 |
-
|
110 |
-
|
111 |
-
def form_expert(
|
112 |
-
model_active_params: float,
|
113 |
-
model_total_params: float,
|
114 |
-
prompt_generated_tokens: int,
|
115 |
-
mix_gwp: float,
|
116 |
-
mix_adpe: float,
|
117 |
-
mix_pe: float
|
118 |
-
):
|
119 |
-
impacts = compute_llm_impacts_expert(
|
120 |
-
model_active_parameter_count=model_active_params,
|
121 |
-
model_total_parameter_count=model_total_params,
|
122 |
-
output_token_count=prompt_generated_tokens,
|
123 |
-
request_latency=100000,
|
124 |
-
if_electricity_mix_gwp=mix_gwp,
|
125 |
-
if_electricity_mix_adpe=mix_adpe,
|
126 |
-
if_electricity_mix_pe=mix_pe
|
127 |
-
)
|
128 |
-
return form_output(impacts)
|
129 |
-
|
130 |
|
131 |
CUSTOM = "Custom"
|
132 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
133 |
return CUSTOM
|
134 |
|
|
|
135 |
def model_active_params_fn(model_name: str, n_param: float):
|
136 |
if model_name == CUSTOM:
|
137 |
return n_param
|
@@ -139,6 +87,7 @@ def model_active_params_fn(model_name: str, n_param: float):
|
|
139 |
model = models.find_model(provider=provider, model_name=model_name)
|
140 |
return model.active_parameters or _avg(model.active_parameters_range)
|
141 |
|
|
|
142 |
def model_total_params_fn(model_name: str, n_param: float):
|
143 |
if model_name == CUSTOM:
|
144 |
return n_param
|
@@ -146,82 +95,163 @@ def model_total_params_fn(model_name: str, n_param: float):
|
|
146 |
model = models.find_model(provider=provider, model_name=model_name)
|
147 |
return model.total_parameters or _avg(model.total_parameters_range)
|
148 |
|
|
|
149 |
def mix_fn(country_code: str, mix_adpe: float, mix_pe: float, mix_gwp: float):
|
150 |
if country_code == CUSTOM:
|
151 |
return mix_adpe, mix_pe, mix_gwp
|
152 |
-
return
|
153 |
-
|
154 |
-
with gr.Blocks() as demo:
|
155 |
|
156 |
-
### TITLE
|
157 |
-
gr.Markdown("""
|
158 |
-
# 🌱 EcoLogits Calculator
|
159 |
-
|
160 |
-
**EcoLogits** is a python library that tracks the **energy consumption** and **environmental footprint** of using
|
161 |
-
**generative AI** models through APIs.
|
162 |
|
163 |
-
|
164 |
-
|
165 |
-
✅ Follow us on Linkedin: [GenAI Impact](https://www.linkedin.com/company/genai-impact/posts/?feedView=all)
|
166 |
-
""")
|
167 |
-
|
168 |
-
### SIMPLE CALCULATOR
|
169 |
-
with gr.Tab("Home"):
|
170 |
-
gr.Markdown("""
|
171 |
-
## 😊 Calculator
|
172 |
-
""")
|
173 |
|
|
|
|
|
|
|
174 |
with gr.Row():
|
175 |
-
|
176 |
-
|
177 |
-
label="
|
178 |
-
value=
|
179 |
filterable=True,
|
180 |
-
)
|
181 |
-
prompt = gr.Dropdown(
|
182 |
-
PROMPTS,
|
183 |
-
label="Example prompt",
|
184 |
-
value=50
|
185 |
)
|
186 |
|
187 |
-
|
188 |
-
|
189 |
-
label="
|
190 |
-
|
191 |
-
|
192 |
-
gwp = gr.Markdown(
|
193 |
-
label="gwp",
|
194 |
-
latex_delimiters=[{"left": "$$", "right": "$$", "display": False}]
|
195 |
-
)
|
196 |
-
adpe = gr.Markdown(
|
197 |
-
label="adpe",
|
198 |
-
latex_delimiters=[{"left": "$$", "right": "$$", "display": False}]
|
199 |
)
|
200 |
-
|
201 |
-
|
202 |
-
|
|
|
|
|
|
|
203 |
)
|
204 |
-
|
205 |
-
|
206 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
207 |
with gr.Row():
|
208 |
-
|
209 |
-
label="eq_energy",
|
210 |
-
latex_delimiters=[{"left": "$$", "right": "$$", "display": False}]
|
211 |
-
)
|
212 |
-
equivalent_2 = gr.Markdown(
|
213 |
-
label="eq_gwp",
|
214 |
-
latex_delimiters=[{"left": "$$", "right": "$$", "display": False}]
|
215 |
-
)
|
216 |
-
|
217 |
-
submit_btn = gr.Button("Submit")
|
218 |
-
submit_btn.click(fn=form, inputs=[model, prompt], outputs=[energy, gwp, adpe, pe, equivalent_1, equivalent_2])
|
219 |
-
|
220 |
-
### EXPERT CALCULATOR
|
221 |
-
with gr.Tab("Expert Mode"):
|
222 |
-
gr.Markdown("""
|
223 |
-
## 🤓 Expert mode
|
224 |
-
""")
|
225 |
model = gr.Dropdown(
|
226 |
MODELS + [CUSTOM],
|
227 |
label="Model name",
|
@@ -229,145 +259,140 @@ with gr.Blocks() as demo:
|
|
229 |
filterable=True,
|
230 |
interactive=True
|
231 |
)
|
232 |
-
|
233 |
label="Number of billions of active parameters",
|
234 |
value=45.0,
|
235 |
-
interactive=True
|
236 |
)
|
237 |
-
|
238 |
label="Number of billions of total parameters",
|
239 |
value=45.0,
|
240 |
-
interactive=True
|
241 |
)
|
242 |
-
|
243 |
-
model.change(fn=model_active_params_fn,
|
244 |
-
|
245 |
-
|
246 |
-
|
|
|
|
|
|
|
|
|
247 |
|
248 |
tokens = gr.Number(
|
249 |
-
label="Output tokens",
|
250 |
value=100
|
251 |
)
|
252 |
|
253 |
mix = gr.Dropdown(
|
254 |
-
|
255 |
label="Location",
|
256 |
-
value="WOR",
|
257 |
filterable=True,
|
258 |
interactive=True
|
259 |
)
|
260 |
-
|
261 |
-
label="Electricity mix -
|
262 |
-
value=
|
263 |
interactive=True
|
264 |
)
|
265 |
-
|
266 |
-
label="Electricity mix -
|
267 |
-
value=
|
268 |
interactive=True
|
269 |
)
|
270 |
-
|
271 |
-
label="Electricity mix -
|
272 |
-
value=
|
273 |
interactive=True
|
274 |
)
|
275 |
|
276 |
-
mix.change(fn=mix_fn,
|
277 |
-
|
278 |
-
|
279 |
-
|
280 |
-
|
281 |
-
|
282 |
-
|
283 |
-
|
284 |
-
|
285 |
-
|
286 |
-
|
287 |
-
|
288 |
-
|
289 |
-
|
290 |
-
|
291 |
-
|
292 |
-
|
293 |
-
|
294 |
-
|
295 |
-
|
296 |
-
|
297 |
-
|
298 |
-
|
299 |
-
|
300 |
-
|
301 |
-
|
302 |
-
|
|
|
|
|
|
|
|
|
|
|
303 |
)
|
304 |
-
|
305 |
-
|
306 |
-
|
307 |
-
|
308 |
-
|
309 |
-
|
310 |
-
|
311 |
-
|
312 |
-
|
313 |
-
|
314 |
-
|
315 |
-
|
316 |
-
|
317 |
-
|
318 |
-
|
319 |
-
|
320 |
-
|
321 |
-
|
322 |
-
|
323 |
-
|
324 |
-
|
325 |
-
|
326 |
-
|
327 |
-
|
328 |
-
|
329 |
-
|
330 |
-
|
331 |
-
|
332 |
-
|
333 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
334 |
)
|
335 |
|
336 |
-
|
337 |
-
|
338 |
-
gr.Markdown("""## 📖 Methodology
|
339 |
-
🚧 Under construction
|
340 |
-
""")
|
341 |
-
|
342 |
-
### INFORMATION ABOUT INDICATORS
|
343 |
-
with gr.Accordion("📊 More about the indicators", open = False):
|
344 |
-
gr.Markdown("""
|
345 |
-
- ⚡️ **Energy**: Final energy consumption,
|
346 |
-
- 🌍 **GHG Emissions**: Potential impact on global warming (commonly known as GHG/carbon emissions),
|
347 |
-
- 🪨 **Abiotic Resources**: Impact on the depletion of non-living resources such as minerals or metals,
|
348 |
-
- ⛽️ **Primary Energy**: Total energy consumed from primary sources.
|
349 |
-
""")
|
350 |
-
|
351 |
-
### INFORMATION ABOUT REDUCING IMPACTS
|
352 |
-
with gr.Accordion("📉 How to reduce / limit these impacts ?", open = False):
|
353 |
-
gr.Markdown("""
|
354 |
-
|
355 |
-
* ❓ **Fundamental rule**: Show **sobriety** on the uses of (generative) AI
|
356 |
-
* Questionning the usefulness of the project;
|
357 |
-
* Estimating impacts of the project;
|
358 |
-
* Evaluating the project purpose;
|
359 |
-
* Restricting the use case to the desired purposes.
|
360 |
-
|
361 |
-
* 🦾 On the hardware side
|
362 |
-
* If you can, try to relocate the computing in low emissions and/or energy efficient datacenters.
|
363 |
-
|
364 |
-
* 🤖 On the ML side :
|
365 |
-
* Develop a zero-shot learning approach for general tasks;
|
366 |
-
* Prefer the smaller and yet well-peforming models (using number of parameters for example);
|
367 |
-
* If a specialization is needed, always prefer fine-tuning an existing model than re-training one from scratch;
|
368 |
-
* During model inference, try caching the most popular prompts ("hey, tell me a joke about ...").
|
369 |
-
|
370 |
-
""")
|
371 |
|
372 |
if __name__ == '__main__':
|
373 |
demo.launch()
|
|
|
1 |
+
|
2 |
import gradio as gr
|
|
|
3 |
|
|
|
4 |
|
5 |
from ecologits.tracers.utils import compute_llm_impacts, _avg
|
6 |
from ecologits.impacts.llm import compute_llm_impacts as compute_llm_impacts_expert
|
7 |
from ecologits.impacts.llm import IF_ELECTRICITY_MIX_GWP, IF_ELECTRICITY_MIX_ADPE, IF_ELECTRICITY_MIX_PE
|
8 |
from ecologits.model_repository import models
|
9 |
|
10 |
+
from src.assets import custom_css
|
11 |
+
from src.electricity_mix import COUNTRY_CODES, find_electricity_mix
|
12 |
+
from src.content import (
|
13 |
+
HERO_TEXT,
|
14 |
+
ABOUT_TEXT,
|
15 |
+
CITATION_LABEL,
|
16 |
+
CITATION_TEXT,
|
17 |
+
LICENCE_TEXT, METHODOLOGY_TEXT
|
18 |
+
)
|
19 |
+
from src.constants import (
|
20 |
+
PROVIDERS,
|
21 |
+
OPENAI_MODELS,
|
22 |
+
ANTHROPIC_MODELS,
|
23 |
+
COHERE_MODELS,
|
24 |
+
META_MODELS,
|
25 |
+
MISTRALAI_MODELS,
|
26 |
+
PROMPTS,
|
27 |
+
MODELS
|
28 |
+
)
|
29 |
+
from src.utils import (
|
30 |
+
format_impacts,
|
31 |
+
format_energy_eq_physical_activity,
|
32 |
+
PhysicalActivity,
|
33 |
+
format_energy_eq_electric_vehicle,
|
34 |
+
format_gwp_eq_streaming, format_energy_eq_electricity_production, EnergyProduction,
|
35 |
+
format_gwp_eq_airplane_paris_nyc, format_energy_eq_electricity_consumption_ireland
|
36 |
+
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
37 |
|
38 |
CUSTOM = "Custom"
|
39 |
+
|
40 |
+
|
41 |
+
def model_list(provider: str) -> gr.Dropdown:
|
42 |
+
if provider == "openai":
|
43 |
+
return gr.Dropdown(
|
44 |
+
OPENAI_MODELS,
|
45 |
+
label="Model",
|
46 |
+
value=OPENAI_MODELS[0][1],
|
47 |
+
filterable=True,
|
48 |
+
)
|
49 |
+
elif provider == "anthropic":
|
50 |
+
return gr.Dropdown(
|
51 |
+
ANTHROPIC_MODELS,
|
52 |
+
label="Model",
|
53 |
+
value=ANTHROPIC_MODELS[0][1],
|
54 |
+
filterable=True,
|
55 |
+
)
|
56 |
+
elif provider == "cohere":
|
57 |
+
return gr.Dropdown(
|
58 |
+
COHERE_MODELS,
|
59 |
+
label="Model",
|
60 |
+
value=COHERE_MODELS[0][1],
|
61 |
+
filterable=True,
|
62 |
+
)
|
63 |
+
elif provider == "huggingface_hub/meta":
|
64 |
+
return gr.Dropdown(
|
65 |
+
META_MODELS,
|
66 |
+
label="Model",
|
67 |
+
value=META_MODELS[0][1],
|
68 |
+
filterable=True,
|
69 |
+
)
|
70 |
+
elif provider == "mistralai":
|
71 |
+
return gr.Dropdown(
|
72 |
+
MISTRALAI_MODELS,
|
73 |
+
label="Model",
|
74 |
+
value=MISTRALAI_MODELS[0][1],
|
75 |
+
filterable=True,
|
76 |
+
)
|
77 |
+
|
78 |
+
|
79 |
+
def custom():
|
80 |
return CUSTOM
|
81 |
|
82 |
+
|
83 |
def model_active_params_fn(model_name: str, n_param: float):
|
84 |
if model_name == CUSTOM:
|
85 |
return n_param
|
|
|
87 |
model = models.find_model(provider=provider, model_name=model_name)
|
88 |
return model.active_parameters or _avg(model.active_parameters_range)
|
89 |
|
90 |
+
|
91 |
def model_total_params_fn(model_name: str, n_param: float):
|
92 |
if model_name == CUSTOM:
|
93 |
return n_param
|
|
|
95 |
model = models.find_model(provider=provider, model_name=model_name)
|
96 |
return model.total_parameters or _avg(model.total_parameters_range)
|
97 |
|
98 |
+
|
99 |
def mix_fn(country_code: str, mix_adpe: float, mix_pe: float, mix_gwp: float):
|
100 |
if country_code == CUSTOM:
|
101 |
return mix_adpe, mix_pe, mix_gwp
|
102 |
+
return find_electricity_mix(country_code)
|
|
|
|
|
103 |
|
|
|
|
|
|
|
|
|
|
|
|
|
104 |
|
105 |
+
with gr.Blocks(css=custom_css) as demo:
|
106 |
+
gr.Markdown(HERO_TEXT)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
107 |
|
108 |
+
with gr.Tab("🧮 Calculator"):
|
109 |
+
with gr.Row():
|
110 |
+
gr.Markdown("# Estimate the environmental impacts of LLM inference")
|
111 |
with gr.Row():
|
112 |
+
input_provider = gr.Dropdown(
|
113 |
+
PROVIDERS,
|
114 |
+
label="Provider",
|
115 |
+
value=PROVIDERS[0][1],
|
116 |
filterable=True,
|
|
|
|
|
|
|
|
|
|
|
117 |
)
|
118 |
|
119 |
+
input_model = gr.Dropdown(
|
120 |
+
OPENAI_MODELS,
|
121 |
+
label="Model",
|
122 |
+
value=OPENAI_MODELS[0][1],
|
123 |
+
filterable=True,
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
124 |
)
|
125 |
+
input_provider.change(model_list, input_provider, input_model)
|
126 |
+
|
127 |
+
input_prompt = gr.Dropdown(
|
128 |
+
PROMPTS,
|
129 |
+
label="Example prompt",
|
130 |
+
value=50,
|
131 |
)
|
132 |
+
|
133 |
+
|
134 |
+
@gr.render(inputs=[input_provider, input_model, input_prompt])
|
135 |
+
def render_simple(provider, model, prompt):
|
136 |
+
if provider.startswith("huggingface_hub"):
|
137 |
+
provider = provider.split("/")[0]
|
138 |
+
if models.find_model(provider, model) is not None:
|
139 |
+
impacts = compute_llm_impacts(
|
140 |
+
provider=provider,
|
141 |
+
model_name=model,
|
142 |
+
output_token_count=prompt,
|
143 |
+
request_latency=100000
|
144 |
+
)
|
145 |
+
impacts = format_impacts(impacts)
|
146 |
+
|
147 |
+
# Inference impacts
|
148 |
+
with gr.Blocks():
|
149 |
+
with gr.Row():
|
150 |
+
gr.Markdown("""
|
151 |
+
## Environmental impacts
|
152 |
+
|
153 |
+
To understand how the environmental impacts are computed go to the 📖 Methodology tab.
|
154 |
+
""")
|
155 |
+
with gr.Row():
|
156 |
+
with gr.Column(scale=1, min_width=220):
|
157 |
+
gr.Markdown(f"""
|
158 |
+
<h2 align="center">⚡️ Energy</h2>
|
159 |
+
$$ \Large {impacts.energy.magnitude:.3g} \ \large {impacts.energy.units} $$
|
160 |
+
<p align="center"><i>Evaluates the electricity consumption<i></p><br>
|
161 |
+
""")
|
162 |
+
with gr.Column(scale=1, min_width=220):
|
163 |
+
gr.Markdown(f"""
|
164 |
+
<h2 align="center">🌍️ GHG Emissions</h2>
|
165 |
+
$$ \Large {impacts.gwp.magnitude:.3g} \ \large {impacts.gwp.units} $$
|
166 |
+
<p align="center"><i>Evaluates the effect on global warming<i></p><br>
|
167 |
+
""")
|
168 |
+
with gr.Column(scale=1, min_width=220):
|
169 |
+
gr.Markdown(f"""
|
170 |
+
<h2 align="center">🪨 Abiotic Resources</h2>
|
171 |
+
$$ \Large {impacts.adpe.magnitude:.3g} \ \large {impacts.adpe.units} $$
|
172 |
+
<p align="center"><i>Evaluates the use of metals and minerals<i></p><br>
|
173 |
+
""")
|
174 |
+
with gr.Column(scale=1, min_width=220):
|
175 |
+
gr.Markdown(f"""
|
176 |
+
<h2 align="center">⛽️ Primary Energy</h2>
|
177 |
+
$$ \Large {impacts.pe.magnitude:.3g} \ \large {impacts.pe.units} $$
|
178 |
+
<p align="center"><i>Evaluates the use of energy resources<i></p><br>
|
179 |
+
""")
|
180 |
+
|
181 |
+
# Impacts equivalents
|
182 |
+
with gr.Blocks():
|
183 |
+
with gr.Row():
|
184 |
+
gr.Markdown("""
|
185 |
+
---
|
186 |
+
## That's equivalent to...
|
187 |
+
|
188 |
+
Making this request to the LLM is equivalent to the following actions.
|
189 |
+
""")
|
190 |
+
with gr.Row():
|
191 |
+
physical_activity, distance = format_energy_eq_physical_activity(impacts.energy)
|
192 |
+
if physical_activity == PhysicalActivity.WALKING:
|
193 |
+
physical_activity = "🚶♂️➡️ " + physical_activity.capitalize()
|
194 |
+
if physical_activity == PhysicalActivity.RUNNING:
|
195 |
+
physical_activity = "🏃♂️➡️ " + physical_activity.capitalize()
|
196 |
+
with gr.Column(scale=1, min_width=300):
|
197 |
+
gr.Markdown(f"""
|
198 |
+
<h2 align="center">{physical_activity} $$ \Large {distance.magnitude:.3g}\ {distance.units} $$ </h2>
|
199 |
+
<p align="center"><i>Based on energy consumption<i></p><br>
|
200 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
201 |
+
|
202 |
+
ev_eq = format_energy_eq_electric_vehicle(impacts.energy)
|
203 |
+
with gr.Column(scale=1, min_width=300):
|
204 |
+
gr.Markdown(f"""
|
205 |
+
<h2 align="center">🔋 Electric Vehicle $$ \Large {ev_eq.magnitude:.3g}\ {ev_eq.units} $$ </h2>
|
206 |
+
<p align="center"><i>Based on energy consumption<i></p><br>
|
207 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
208 |
+
|
209 |
+
streaming_eq = format_gwp_eq_streaming(impacts.gwp)
|
210 |
+
with gr.Column(scale=1, min_width=300):
|
211 |
+
gr.Markdown(f"""
|
212 |
+
<h2 align="center">⏯️ Streaming $$ \Large {streaming_eq.magnitude:.3g}\ {streaming_eq.units} $$ </h2>
|
213 |
+
<p align="center"><i>Based on GHG emissions<i></p><br>
|
214 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
215 |
+
|
216 |
+
# Bigger scale impacts equivalent
|
217 |
+
with gr.Blocks():
|
218 |
+
with gr.Row():
|
219 |
+
gr.Markdown("""
|
220 |
+
## What if 1% of the planet does this everyday for 1 year?
|
221 |
+
|
222 |
+
If this use case is largely deployed around the world the equivalent impacts would be.
|
223 |
+
""")
|
224 |
+
with gr.Row():
|
225 |
+
electricity_production, count = format_energy_eq_electricity_production(impacts.energy)
|
226 |
+
if electricity_production == EnergyProduction.NUCLEAR:
|
227 |
+
emoji = "☢️"
|
228 |
+
name = "Nuclear power plants"
|
229 |
+
if electricity_production == EnergyProduction.WIND:
|
230 |
+
emoji = "💨️ "
|
231 |
+
name = "Wind turbines"
|
232 |
+
with gr.Column(scale=1, min_width=300):
|
233 |
+
gr.Markdown(f"""
|
234 |
+
<h2 align="center">{emoji} $$ \Large {count:.0f} $$ {name} <span style="font-size: 12px">(yearly)</span></h2>
|
235 |
+
<p align="center"><i>Based on electricity consumption<i></p><br>
|
236 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
237 |
+
|
238 |
+
ireland_count = format_energy_eq_electricity_consumption_ireland(impacts.energy)
|
239 |
+
with gr.Column(scale=1, min_width=300):
|
240 |
+
gr.Markdown(f"""
|
241 |
+
<h2 align="center">🇮🇪 $$ \Large {ireland_count:.2g} $$ x Ireland <span style="font-size: 12px">(yearly ⚡️ cons.)</span></h2>
|
242 |
+
<p align="center"><i>Based on electricity consumption<i></p><br>
|
243 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
244 |
+
|
245 |
+
paris_nyc_airplane = format_gwp_eq_airplane_paris_nyc(impacts.gwp)
|
246 |
+
with gr.Column(scale=1, min_width=300):
|
247 |
+
gr.Markdown(f"""
|
248 |
+
<h2 align="center">✈️ $$ \Large {paris_nyc_airplane:,.0f} $$ Paris ↔ NYC </h2>
|
249 |
+
<p align="center"><i>Based on GHG emissions<i></p><br>
|
250 |
+
""", latex_delimiters=[{"left": "$$", "right": "$$", "display": False}])
|
251 |
+
|
252 |
+
with gr.Tab("🤓 Expert Mode"):
|
253 |
with gr.Row():
|
254 |
+
gr.Markdown("# 🤓 Expert mode")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
255 |
model = gr.Dropdown(
|
256 |
MODELS + [CUSTOM],
|
257 |
label="Model name",
|
|
|
259 |
filterable=True,
|
260 |
interactive=True
|
261 |
)
|
262 |
+
input_model_active_params = gr.Number(
|
263 |
label="Number of billions of active parameters",
|
264 |
value=45.0,
|
265 |
+
interactive=True
|
266 |
)
|
267 |
+
input_model_total_params = gr.Number(
|
268 |
label="Number of billions of total parameters",
|
269 |
value=45.0,
|
270 |
+
interactive=True
|
271 |
)
|
272 |
+
|
273 |
+
model.change(fn=model_active_params_fn,
|
274 |
+
inputs=[model, input_model_active_params],
|
275 |
+
outputs=[input_model_active_params])
|
276 |
+
model.change(fn=model_total_params_fn,
|
277 |
+
inputs=[model, input_model_total_params],
|
278 |
+
outputs=[input_model_total_params])
|
279 |
+
input_model_active_params.input(fn=custom, outputs=[model])
|
280 |
+
input_model_total_params.input(fn=custom, outputs=[model])
|
281 |
|
282 |
tokens = gr.Number(
|
283 |
+
label="Output tokens",
|
284 |
value=100
|
285 |
)
|
286 |
|
287 |
mix = gr.Dropdown(
|
288 |
+
COUNTRY_CODES + [CUSTOM],
|
289 |
label="Location",
|
290 |
+
value="WOR",
|
291 |
filterable=True,
|
292 |
interactive=True
|
293 |
)
|
294 |
+
input_mix_gwp = gr.Number(
|
295 |
+
label="Electricity mix - GHG emissions [kgCO2eq / kWh]",
|
296 |
+
value=IF_ELECTRICITY_MIX_GWP,
|
297 |
interactive=True
|
298 |
)
|
299 |
+
input_mix_adpe = gr.Number(
|
300 |
+
label="Electricity mix - Abiotic resources [kgSbeq / kWh]",
|
301 |
+
value=IF_ELECTRICITY_MIX_ADPE,
|
302 |
interactive=True
|
303 |
)
|
304 |
+
input_mix_pe = gr.Number(
|
305 |
+
label="Electricity mix - Primary energy [MJ / kWh]",
|
306 |
+
value=IF_ELECTRICITY_MIX_PE,
|
307 |
interactive=True
|
308 |
)
|
309 |
|
310 |
+
mix.change(fn=mix_fn,
|
311 |
+
inputs=[mix, input_mix_gwp, input_mix_adpe, input_mix_pe],
|
312 |
+
outputs=[input_mix_gwp, input_mix_adpe, input_mix_pe])
|
313 |
+
input_mix_gwp.input(fn=custom, outputs=mix)
|
314 |
+
input_mix_adpe.input(fn=custom, outputs=mix)
|
315 |
+
input_mix_pe.input(fn=custom, outputs=mix)
|
316 |
+
|
317 |
+
|
318 |
+
@gr.render(inputs=[
|
319 |
+
input_model_active_params,
|
320 |
+
input_model_total_params,
|
321 |
+
input_prompt,
|
322 |
+
input_mix_gwp,
|
323 |
+
input_mix_adpe,
|
324 |
+
input_mix_pe
|
325 |
+
])
|
326 |
+
def render_expert(
|
327 |
+
model_active_params,
|
328 |
+
model_total_params,
|
329 |
+
prompt,
|
330 |
+
mix_gwp,
|
331 |
+
mix_adpe,
|
332 |
+
mix_pe
|
333 |
+
):
|
334 |
+
impacts = compute_llm_impacts_expert(
|
335 |
+
model_active_parameter_count=model_active_params,
|
336 |
+
model_total_parameter_count=model_total_params,
|
337 |
+
output_token_count=prompt,
|
338 |
+
request_latency=100000,
|
339 |
+
if_electricity_mix_gwp=mix_gwp,
|
340 |
+
if_electricity_mix_adpe=mix_adpe,
|
341 |
+
if_electricity_mix_pe=mix_pe
|
342 |
)
|
343 |
+
impacts = format_impacts(impacts)
|
344 |
+
|
345 |
+
with gr.Blocks():
|
346 |
+
with gr.Row():
|
347 |
+
gr.Markdown("## Environmental impacts")
|
348 |
+
with gr.Row():
|
349 |
+
with gr.Column(scale=1, min_width=220):
|
350 |
+
gr.Markdown(f"""
|
351 |
+
<h2 align="center">⚡️ Energy</h2>
|
352 |
+
$$ \Large {impacts.energy.magnitude:.3g} \ \large {impacts.energy.units} $$
|
353 |
+
<p align="center"><i>Evaluates the electricity consumption<i></p><br>
|
354 |
+
""")
|
355 |
+
with gr.Column(scale=1, min_width=220):
|
356 |
+
gr.Markdown(f"""
|
357 |
+
<h2 align="center">🌍️ GHG Emissions</h2>
|
358 |
+
$$ \Large {impacts.gwp.magnitude:.3g} \ \large {impacts.gwp.units} $$
|
359 |
+
<p align="center"><i>Evaluates the effect on global warming<i></p><br>
|
360 |
+
""")
|
361 |
+
with gr.Column(scale=1, min_width=220):
|
362 |
+
gr.Markdown(f"""
|
363 |
+
<h2 align="center">🪨 Abiotic Resources</h2>
|
364 |
+
$$ \Large {impacts.adpe.magnitude:.3g} \ \large {impacts.adpe.units} $$
|
365 |
+
<p align="center"><i>Evaluates the use of metals and minerals<i></p><br>
|
366 |
+
""")
|
367 |
+
with gr.Column(scale=1, min_width=220):
|
368 |
+
gr.Markdown(f"""
|
369 |
+
<h2 align="center">⛽️ Primary Energy</h2>
|
370 |
+
$$ \Large {impacts.pe.magnitude:.3g} \ \large {impacts.pe.units} $$
|
371 |
+
<p align="center"><i>Evaluates the use of energy resources<i></p><br>
|
372 |
+
""")
|
373 |
+
|
374 |
+
with gr.Tab("📖 Methodology"):
|
375 |
+
gr.Markdown(METHODOLOGY_TEXT,
|
376 |
+
elem_classes="descriptive-text",
|
377 |
+
latex_delimiters=[
|
378 |
+
{"left": "$$", "right": "$$", "display": True},
|
379 |
+
{"left": "$", "right": "$", "display": False}
|
380 |
+
])
|
381 |
+
|
382 |
+
with gr.Tab("ℹ️ About"):
|
383 |
+
gr.Markdown(ABOUT_TEXT, elem_classes="descriptive-text",)
|
384 |
+
|
385 |
+
with gr.Accordion("📚 Citation", open=False):
|
386 |
+
gr.Textbox(
|
387 |
+
value=CITATION_TEXT,
|
388 |
+
label=CITATION_LABEL,
|
389 |
+
interactive=False,
|
390 |
+
show_copy_button=True,
|
391 |
+
lines=len(CITATION_TEXT.split('\n')),
|
392 |
)
|
393 |
|
394 |
+
# License
|
395 |
+
gr.Markdown(LICENCE_TEXT)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
396 |
|
397 |
if __name__ == '__main__':
|
398 |
demo.launch()
|
src/assets.py
ADDED
@@ -0,0 +1,6 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
custom_css = """
|
2 |
+
|
3 |
+
.descriptive-text span {
|
4 |
+
font-size: 16px !important;
|
5 |
+
}
|
6 |
+
"""
|
src/constants.py
ADDED
@@ -0,0 +1,82 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
PROVIDERS = [
|
2 |
+
("OpenAI", "openai"),
|
3 |
+
("Anthropic", "anthropic"),
|
4 |
+
("Cohere", "cohere"),
|
5 |
+
("Meta", "huggingface_hub/meta"),
|
6 |
+
("Mistral AI", "mistralai"),
|
7 |
+
]
|
8 |
+
|
9 |
+
OPENAI_MODELS = [
|
10 |
+
("GPT-3.5-Turbo", "gpt-3.5-turbo"),
|
11 |
+
("GPT-4", "gpt-4"),
|
12 |
+
]
|
13 |
+
|
14 |
+
ANTHROPIC_MODELS = [
|
15 |
+
("Claude 3 Opus", "claude-3-opus-20240229"),
|
16 |
+
("Claude 3 Sonnet", "claude-3-sonnet-20240229"),
|
17 |
+
("Claude 3 Haiku", "claude-3-haiku-20240307"),
|
18 |
+
("Claude 2.1", "claude-2.1"),
|
19 |
+
("Claude 2.0", "claude-2.0"),
|
20 |
+
("Claude Instant 1.2", "claude-instant-1.2"),
|
21 |
+
]
|
22 |
+
|
23 |
+
COHERE_MODELS = [
|
24 |
+
("Command Light", "command-light"),
|
25 |
+
("Command", "command"),
|
26 |
+
("Command R", "command-r"),
|
27 |
+
("Command R+", "command-r-plus"),
|
28 |
+
]
|
29 |
+
|
30 |
+
META_MODELS = [
|
31 |
+
("Llama 3 8B", "meta-llama/Meta-Llama-3-8B"),
|
32 |
+
("Llama 3 70B", "meta-llama/Meta-Llama-3-70B"),
|
33 |
+
("Llama 2 7B", "meta-llama/Llama-2-7b-hf"),
|
34 |
+
("Llama 2 13B", "meta-llama/Llama-2-13b-hf"),
|
35 |
+
("Llama 2 70B", "meta-llama/Llama-2-70b-hf"),
|
36 |
+
]
|
37 |
+
|
38 |
+
MISTRALAI_MODELS = [
|
39 |
+
("Mistral 7B", "open-mistral-7b"),
|
40 |
+
("Mixtral 8x7B", "open-mixtral-8x7b"),
|
41 |
+
("Mixtral 8x22B", "open-mixtral-8x22b"),
|
42 |
+
("Tiny", "mistral-tiny-2312"),
|
43 |
+
("Small", "mistral-small-2402"),
|
44 |
+
("Medium", "mistral-medium-2312"),
|
45 |
+
("Large", "mistral-large-2402"),
|
46 |
+
]
|
47 |
+
|
48 |
+
PROMPTS = [
|
49 |
+
("Write a Tweet", 50),
|
50 |
+
("Write an email", 170),
|
51 |
+
("Write an article summary", 250),
|
52 |
+
("Small conversation with a chatbot", 400),
|
53 |
+
("Write a report of 5 pages", 5000),
|
54 |
+
]
|
55 |
+
PROMPTS = [(s + f" ({v} output tokens)", v) for (s, v) in PROMPTS]
|
56 |
+
|
57 |
+
MODELS = [
|
58 |
+
("OpenAI / GPT-3.5-Turbo", "openai/gpt-3.5-turbo"),
|
59 |
+
("OpenAI / GPT-4", "openai/gpt-4"),
|
60 |
+
("Anthropic / Claude 3 Opus", "anthropic/claude-3-opus-20240229"),
|
61 |
+
("Anthropic / Claude 3 Sonnet", "anthropic/claude-3-sonnet-20240229"),
|
62 |
+
("Anthropic / Claude 3 Haiku", "anthropic/claude-3-haiku-20240307"),
|
63 |
+
("Anthropic / Claude 2.1", "anthropic/claude-2.1"),
|
64 |
+
("Anthropic / Claude 2.0", "anthropic/claude-2.0"),
|
65 |
+
("Anthropic / Claude Instant 1.2", "anthropic/claude-instant-1.2"),
|
66 |
+
("Mistral AI / Mistral 7B", "mistralai/open-mistral-7b"),
|
67 |
+
("Mistral AI / Mixtral 8x7B", "mistralai/open-mixtral-8x7b"),
|
68 |
+
("Mistral AI / Mixtral 8x22B", "mistralai/open-mixtral-8x22b"),
|
69 |
+
("Mistral AI / Tiny", "mistralai/mistral-tiny-2312"),
|
70 |
+
("Mistral AI / Small", "mistralai/mistral-small-2402"),
|
71 |
+
("Mistral AI / Medium", "mistralai/mistral-medium-2312"),
|
72 |
+
("Mistral AI / Large", "mistralai/mistral-large-2402"),
|
73 |
+
("Meta / Llama 3 8B", "huggingface_hub/meta-llama/Meta-Llama-3-8B"),
|
74 |
+
("Meta / Llama 3 70B", "huggingface_hub/meta-llama/Meta-Llama-3-70B"),
|
75 |
+
("Meta / Llama 2 7B", "huggingface_hub/meta-llama/Llama-2-7b-hf"),
|
76 |
+
("Meta / Llama 2 13B", "huggingface_hub/meta-llama/Llama-2-13b-hf"),
|
77 |
+
("Meta / Llama 2 70B", "huggingface_hub/meta-llama/Llama-2-70b-hf"),
|
78 |
+
("Cohere / Command Light", "cohere/command-light"),
|
79 |
+
("Cohere / Command", "cohere/command"),
|
80 |
+
("Cohere / Command R", "cohere/command-r"),
|
81 |
+
("Cohere / Command R+", "cohere/command-r-plus"),
|
82 |
+
]
|
src/content.py
ADDED
@@ -0,0 +1,229 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
HERO_TEXT = """
|
3 |
+
<div align="center">
|
4 |
+
<a href="https://ecologits.ai/">
|
5 |
+
<img style="max-height: 80px" alt="EcoLogits" src="https://raw.githubusercontent.com/genai-impact/ecologits/main/docs/assets/logo_light.png">
|
6 |
+
</a>
|
7 |
+
</div>
|
8 |
+
|
9 |
+
<h1 align="center">🧮 EcoLogits Calculator</h1>
|
10 |
+
<div align="center">
|
11 |
+
<p style="max-width: 500px; text-align: center">
|
12 |
+
<i><b>EcoLogits</b> is a python library that tracks the <b>energy consumption</b> and <b>environmental
|
13 |
+
footprint</b> of using <b>generative AI</b> models through APIs.</i>
|
14 |
+
</p>
|
15 |
+
</div>
|
16 |
+
<br>
|
17 |
+
|
18 |
+
This tool is developed and maintained by [GenAI Impact](https://genai-impact.org/) non-profit. Learn more about
|
19 |
+
🌱 EcoLogits by reading the documentation on [ecologits.ai](https://ecologits.ai).
|
20 |
+
|
21 |
+
🩷 Support us by giving a ⭐️ on our [GitHub repository](https://github.com/genai-impact/ecologits) and by following our [LinkedIn page](https://www.linkedin.com/company/genai-impact/).
|
22 |
+
"""
|
23 |
+
|
24 |
+
ABOUT_TEXT = r"""
|
25 |
+
## 🎯 Our goal
|
26 |
+
|
27 |
+
**The main goal of the EcoLogits Calculator is to raise awareness on the environmental impacts of LLM inference.**
|
28 |
+
|
29 |
+
The rapid evolution of generative AI is reshaping numerous industries and aspects of our daily lives. While these
|
30 |
+
advancements offer some benefits, they also **pose substantial environmental challenges that cannot be overlooked**.
|
31 |
+
Plus the issue of AI's environmental footprint as been mainly discussed at training stage but rarely at the inference
|
32 |
+
stage. That is an issue because **inference impacts for LLMs can largely overcome the training impacts when deployed
|
33 |
+
at large scales**.
|
34 |
+
|
35 |
+
At **[GenAI Impact](https://genai-impact.org/) we are dedicated to understanding and mitigating the environmental
|
36 |
+
impacts of generative AI** through rigorous research, innovative tools, and community engagement. Especially, in early
|
37 |
+
2024 we have launched an new open-source tool called [EcoLogits](https://github.com/genai-impact/ecologits) that tracks
|
38 |
+
the energy consumption and environmental footprint of using generative AI models through APIs.
|
39 |
+
|
40 |
+
## 🙋 FAQ
|
41 |
+
|
42 |
+
**Which generative AI models or providers are supported?**
|
43 |
+
|
44 |
+
To see the full list of **generative AI providers** currently supported by EcoLogits, see the following
|
45 |
+
[documentation page](https://ecologits.ai/providers/). As of today we only support LLMs but we plan to add support for
|
46 |
+
embeddings, image generation, multi-modal models and more. If you are interested don't hesitate to
|
47 |
+
[join us](https://genai-impact.org/contact/) and accelerate our work!
|
48 |
+
|
49 |
+
**How to reduce AI environmental impacts?**
|
50 |
+
|
51 |
+
* Look at **indirect impacts** of your project. Does the finality of your project is impacting negatively the
|
52 |
+
environment?
|
53 |
+
* **Be frugal** and question your usage or need of AI
|
54 |
+
* Do you really need AI to solve your problem?
|
55 |
+
* Do you really need GenAI to solve your problem? (you can read this [paper](https://aclanthology.org/2023.emnlp-industry.39.pdf))
|
56 |
+
* Use small and specialized models to solve your problem.
|
57 |
+
* Evaluate before, during and after the development of your project the environmental impacts with tools like
|
58 |
+
🌱 [EcoLogits](https://github.com/genai-impact/ecologits) or [CodeCarbon](https://github.com/mlco2/codecarbon)
|
59 |
+
(see [more tools](https://github.com/samuelrince/awesome-green-ai))
|
60 |
+
* Restrict the use case and limit the usage of your tool or feature to the desired purpose.
|
61 |
+
* Do NOT buy new GPUs / hardware
|
62 |
+
* Hardware manufacturing for data centers is around 50% of the impact.
|
63 |
+
* Use cloud instances that are located in low emissions / high energy efficiency data centers
|
64 |
+
(see [electricitymaps.com](https://app.electricitymaps.com/map))
|
65 |
+
* Optimize your models for production
|
66 |
+
* Quantize your models.
|
67 |
+
* Use inference optimization tricks.
|
68 |
+
* Prefer fine-tuning of small and existing models over generalist models.
|
69 |
+
|
70 |
+
**What is the difference between **EcoLogits** and [CodeCarbon](https://github.com/mlco2/codecarbon)?**
|
71 |
+
|
72 |
+
EcoLogits is focused on estimating the environmental impacts of generative AI (only LLMs for now) used **through API
|
73 |
+
providers (such as OpenAI, Anthropic, Cloud APIs...)** whereas CodeCarbon is more general tool to measure energy
|
74 |
+
consumption and estimate GHG emissions measurement. If you deploy LLMs locally we encourage you to use CodeCarbon to
|
75 |
+
get real numbers of your energy consumption.
|
76 |
+
|
77 |
+
## 🤗 Contributing
|
78 |
+
|
79 |
+
We are eager to get feedback from the community, don't hesitate to engage the discussion with us on this
|
80 |
+
[GitHub thread](https://github.com/genai-impact/ecologits/discussions/45) or message us on
|
81 |
+
[LinkedIn](https://www.linkedin.com/company/genai-impact/).
|
82 |
+
|
83 |
+
We also welcome any open-source contributions on 🌱 **[EcoLogits](https://github.com/genai-impact/ecologits)** or on
|
84 |
+
🧮 **EcoLogits Calculator**.
|
85 |
+
|
86 |
+
## ⚖️ License
|
87 |
+
|
88 |
+
<p xmlns:cc="http://creativecommons.org/ns#" >
|
89 |
+
This work is licensed under
|
90 |
+
<a href="https://creativecommons.org/licenses/by-sa/4.0/?ref=chooser-v1" target="_blank" rel="license noopener noreferrer" style="display:inline-block;">
|
91 |
+
CC BY-SA 4.0
|
92 |
+
</a>
|
93 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/cc.svg?ref=chooser-v1" alt="">
|
94 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/by.svg?ref=chooser-v1" alt="">
|
95 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/sa.svg?ref=chooser-v1" alt="">
|
96 |
+
</p>
|
97 |
+
|
98 |
+
## 🙌 Acknowledgement
|
99 |
+
|
100 |
+
We thank [Data For Good](https://dataforgood.fr/) and [Boavizta](https://boavizta.org/en) for supporting the
|
101 |
+
development of this project. Their contributions of tools, best practices, and expertise in environmental impact
|
102 |
+
assessment have been invaluable.
|
103 |
+
|
104 |
+
We also extend our gratitude to the open-source contributions of 🤗 [Hugging Face](git.lnyan.comm) on the LLM-Perf
|
105 |
+
Leaderboard.
|
106 |
+
|
107 |
+
## 🤝 Contact
|
108 |
+
|
109 |
+
For general question on the project, please use the [GitHub thread](https://github.com/genai-impact/ecologits/discussions/45).
|
110 |
+
Otherwise use our contact form on [genai-impact.org/contact](https://genai-impact.org/contact/).
|
111 |
+
"""
|
112 |
+
|
113 |
+
|
114 |
+
METHODOLOGY_TEXT = r"""
|
115 |
+
## 📖 Methodology
|
116 |
+
|
117 |
+
We have developed a methodology to **estimate the energy consumption and environmental impacts for an LLM inference**
|
118 |
+
based on request parameters and hypotheses on the data center location, the hardware used, the model architecture and
|
119 |
+
more.
|
120 |
+
|
121 |
+
In this section we will only cover the principles of the methodology related to the 🧮 **EcoLogits Calculator**. If
|
122 |
+
you wish to learn more on the environmental impacts modeling of an LLM request checkout the
|
123 |
+
🌱 [EcoLogits documentation page](https://ecologits.ai/methodology/).
|
124 |
+
|
125 |
+
### Modeling impacts of an LLM request
|
126 |
+
|
127 |
+
The environmental impacts of an LLM inference are split into the **usage impacts** $I_{request}^u$ to account for
|
128 |
+
electricity consumption and the **embodied impacts** $I_{request}^e$ that relates to resource extraction, hardware
|
129 |
+
manufacturing and transportation. In general terms it can be expressed as follow:
|
130 |
+
|
131 |
+
$$ I_{request} = I_{request}^u + I_{request}^e $$
|
132 |
+
|
133 |
+
$$ I_{request} = E_{request}*F_{em}+\frac{\Delta T}{\Delta L}*I_{server}^e $$
|
134 |
+
|
135 |
+
With,
|
136 |
+
|
137 |
+
* $E_{request}$ the estimated energy consumption of the server and its cooling system.
|
138 |
+
* $F_{em}$ the electricity mix that depends on the country and time.
|
139 |
+
* $\frac{\Delta T}{\Delta L}$ the hardware usage ratio i.e. the computation time over the lifetime of the hardware.
|
140 |
+
* $I_{server}^e$ the embodied impacts of the server.
|
141 |
+
|
142 |
+
Additionally, to ⚡️ **direct energy consumption** the environmental impacts are expressed in **three dimensions
|
143 |
+
(multi-criteria impacts)** that are:
|
144 |
+
|
145 |
+
* 🌍 **Global Warming Potential** (GWP): Potential impact on global warming in kgCO2eq (commonly known as GHG/carbon
|
146 |
+
emissions).
|
147 |
+
* 🪨 **Abiotic Depletion Potential for Elements** (ADPe): Impact on the depletion of non-living resources such as
|
148 |
+
minerals or metals in kgSbeq.
|
149 |
+
* ⛽️ **Primary Energy** (PE): Total energy consumed from primary sources in MJ.
|
150 |
+
|
151 |
+
### Principles, Data and Hypotheses
|
152 |
+
|
153 |
+
We use a **bottom-up methodology** to model impacts, meaning that we will estimate the impacts of low-level physical
|
154 |
+
components to then estimate the impacts at software level (in that case an LLM inference). We also rely on **Life
|
155 |
+
Cycle Approach (LCA) proxies and approach** to model both usage and embodied phases with multi-criteria impacts.
|
156 |
+
If you are interested in this approach we recommend you to read the following [Boavizta](https://boavizta.org/)
|
157 |
+
resources.
|
158 |
+
|
159 |
+
* [Digital & environment: How to evaluate server manufacturing footprint, beyond greenhouse gas emissions?](https://boavizta.org/en/blog/empreinte-de-la-fabrication-d-un-serveur)
|
160 |
+
* [Boavizta API automated evaluation of environmental impacts of ICT services and equipments](https://boavizta.org/en/blog/boavizta-api-automated-evaluation-of-ict-impacts-on-the-environment)
|
161 |
+
* [Boavizta API documentation](https://doc.api.boavizta.org/)
|
162 |
+
|
163 |
+
We leverage **open data to estimate the environmental impacts**, here is an exhaustive list of our data providers.
|
164 |
+
|
165 |
+
* [LLM-Perf Leaderboard](https://huggingface.co/spaces/optimum/llm-perf-leaderboard) to estimate GPU energy consumption
|
166 |
+
and latency based on the model architecture and number of output tokens.
|
167 |
+
* [Boavizta API](https://github.com/Boavizta/boaviztapi) to estimate server embodied impacts and base energy
|
168 |
+
consumption.
|
169 |
+
* [ADEME Base Empreinte®](https://base-empreinte.ademe.fr/) for electricity mix impacts per country.
|
170 |
+
|
171 |
+
Finally here are the **main hypotheses** we have made to compute the impacts.
|
172 |
+
|
173 |
+
* ⚠️ **We *"guesstimate"* the model architecture of proprietary LLMs when not disclosed by the provider.**
|
174 |
+
* Production setup: quantized models running on data center grade servers and GPUs such as A100.
|
175 |
+
* Electricity mix does not depend on time (help us enhance EcoLogits and work on this [issue](https://github.com/genai-impact/ecologits/issues/42))
|
176 |
+
* Ignore the following impacts: unused cloud resources, data center building, network and end-user devices... (for now)
|
177 |
+
|
178 |
+
## Equivalents
|
179 |
+
|
180 |
+
### 🚶♂️➡️ Walking or 🏃♂️➡️ running distance
|
181 |
+
|
182 |
+
🔴 TODO
|
183 |
+
|
184 |
+
### 🔋 Electric Vehicle distance
|
185 |
+
|
186 |
+
🔴 TODO
|
187 |
+
|
188 |
+
### ⏯️ Streaming time
|
189 |
+
|
190 |
+
🔴 TODO
|
191 |
+
|
192 |
+
### Number of 💨 wind turbines or ☢️ nuclear plants
|
193 |
+
|
194 |
+
🔴 TODO
|
195 |
+
|
196 |
+
### Multiplier of 🇮🇪 Ireland electricity consumption
|
197 |
+
|
198 |
+
🔴 TODO
|
199 |
+
|
200 |
+
### Number of ✈️ Paris ↔ New York City flights
|
201 |
+
|
202 |
+
🔴 TODO
|
203 |
+
|
204 |
+
**If you are motivated to help us test and enhance this methodology
|
205 |
+
[contact us](https://genai-impact.org/contact/)!** 💪
|
206 |
+
"""
|
207 |
+
|
208 |
+
CITATION_LABEL = "BibTeX citation for EcoLogits Calculator and the EcoLogits library:"
|
209 |
+
CITATION_TEXT = r"""@misc{ecologits-calculator,
|
210 |
+
author={Samuel Rincé, Adrien Banse and Valentin Defour},
|
211 |
+
title={EcoLogits Calculator},
|
212 |
+
year={2024},
|
213 |
+
howpublished= {\url{https://huggingface.co/spaces/genai-impact/ecologits-calculator}},
|
214 |
+
}
|
215 |
+
@software{ecologits,
|
216 |
+
author = {Samuel Rincé, Adrien Banse, Vinh Nguyen and Luc Berton},
|
217 |
+
publisher = {GenAI Impact},
|
218 |
+
title = {EcoLogits: track the energy consumption and environmental footprint of using generative AI models through APIs.},
|
219 |
+
}"""
|
220 |
+
|
221 |
+
LICENCE_TEXT = """<p xmlns:cc="http://creativecommons.org/ns#" >
|
222 |
+
This work is licensed under
|
223 |
+
<a href="https://creativecommons.org/licenses/by-sa/4.0/?ref=chooser-v1" target="_blank" rel="license noopener noreferrer" style="display:inline-block;">
|
224 |
+
CC BY-SA 4.0
|
225 |
+
</a>
|
226 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/cc.svg?ref=chooser-v1" alt="">
|
227 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/by.svg?ref=chooser-v1" alt="">
|
228 |
+
<img style="display:inline-block;height:22px!important;margin-left:3px;vertical-align:text-bottom;" src="https://mirrors.creativecommons.org/presskit/icons/sa.svg?ref=chooser-v1" alt="">
|
229 |
+
</p>"""
|
{data → src}/electricity_mix.csv
RENAMED
File without changes
|
data/mixes.py → src/electricity_mix.py
RENAMED
@@ -1,7 +1,7 @@
|
|
1 |
from csv import DictReader
|
2 |
|
3 |
PATH = "data/electricity_mix.csv"
|
4 |
-
|
5 |
("🌎 World", "WOR"),
|
6 |
("🇪🇺 Europe", "EEE"),
|
7 |
("🇿🇼 Zimbabwe", "ZWE"),
|
@@ -143,7 +143,8 @@ MIXES = [
|
|
143 |
("🇦🇪 United Arab Emirates", "ARE")
|
144 |
]
|
145 |
|
146 |
-
|
|
|
147 |
# TODO: Maybe more optimal to construct database at the beginning of the app
|
148 |
# in the same fashion as find_model
|
149 |
res = []
|
@@ -151,4 +152,4 @@ def find_mix(code: str):
|
|
151 |
csv = DictReader(fd)
|
152 |
for row in csv:
|
153 |
res += [float(row[code])]
|
154 |
-
return res
|
|
|
1 |
from csv import DictReader
|
2 |
|
3 |
PATH = "data/electricity_mix.csv"
|
4 |
+
COUNTRY_CODES = [
|
5 |
("🌎 World", "WOR"),
|
6 |
("🇪🇺 Europe", "EEE"),
|
7 |
("🇿🇼 Zimbabwe", "ZWE"),
|
|
|
143 |
("🇦🇪 United Arab Emirates", "ARE")
|
144 |
]
|
145 |
|
146 |
+
|
147 |
+
def find_electricity_mix(code: str):
|
148 |
# TODO: Maybe more optimal to construct database at the beginning of the app
|
149 |
# in the same fashion as find_model
|
150 |
res = []
|
|
|
152 |
csv = DictReader(fd)
|
153 |
for row in csv:
|
154 |
res += [float(row[code])]
|
155 |
+
return res
|
src/utils.py
ADDED
@@ -0,0 +1,182 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from dataclasses import dataclass
|
2 |
+
from enum import Enum
|
3 |
+
|
4 |
+
from ecologits.impacts.models import Impacts, Energy, GWP, ADPe, PE
|
5 |
+
from pint import UnitRegistry, Quantity
|
6 |
+
|
7 |
+
|
8 |
+
u = UnitRegistry()
|
9 |
+
u.define('Wh = watt_hour')
|
10 |
+
u.define('kWh = kilowatt_hour')
|
11 |
+
u.define('MWh = megawatt_hour')
|
12 |
+
u.define('GWh = gigawatt_hour')
|
13 |
+
u.define('TWh = terawatt_hour')
|
14 |
+
u.define('gCO2eq = gram')
|
15 |
+
u.define('kgCO2eq = kilogram')
|
16 |
+
u.define('tCO2eq = metricton')
|
17 |
+
u.define('kgSbeq = kilogram')
|
18 |
+
u.define('kJ = kilojoule')
|
19 |
+
u.define('MJ = megajoule')
|
20 |
+
u.define('m = meter')
|
21 |
+
u.define('km = kilometer')
|
22 |
+
u.define('s = second')
|
23 |
+
u.define('min = minute')
|
24 |
+
u.define('h = hour')
|
25 |
+
q = u.Quantity
|
26 |
+
|
27 |
+
|
28 |
+
@dataclass
|
29 |
+
class QImpacts:
|
30 |
+
energy: Quantity
|
31 |
+
gwp: Quantity
|
32 |
+
adpe: Quantity
|
33 |
+
pe: Quantity
|
34 |
+
|
35 |
+
|
36 |
+
class PhysicalActivity(str, Enum):
|
37 |
+
RUNNING = "running"
|
38 |
+
WALKING = "walking"
|
39 |
+
|
40 |
+
|
41 |
+
class EnergyProduction(str, Enum):
|
42 |
+
NUCLEAR = "nuclear"
|
43 |
+
WIND = "wind"
|
44 |
+
|
45 |
+
|
46 |
+
COUNTRIES = [
|
47 |
+
("cook_islands", 38.81, 9_556),
|
48 |
+
("tonga", 51.15, 104_490),
|
49 |
+
("comoros", 100, 821_632),
|
50 |
+
("samoa", 100, 821_632),
|
51 |
+
]
|
52 |
+
|
53 |
+
|
54 |
+
# From https://www.runningtools.com/energyusage.htm
|
55 |
+
RUNNING_ENERGY_EQ = q("294 kJ / km") # running 1 km at 10 km/h with a weight of 70 kg
|
56 |
+
WALKING_ENERGY_EQ = q("196 kJ / km") # walking 1 km at 3 km/h with a weight of 70 kg
|
57 |
+
|
58 |
+
# From https://selectra.info/energie/actualites/insolite/consommation-vehicules-electriques-france-2040
|
59 |
+
# and https://www.tesla.com/fr_fr/support/power-consumption
|
60 |
+
EV_ENERGY_EQ = q("0.17 kWh / km")
|
61 |
+
|
62 |
+
# From https://impactco2.fr/outils/comparateur?value=1&comparisons=streamingvideo
|
63 |
+
STREAMING_GWP_EQ = q("15.6 h / kgCO2eq")
|
64 |
+
|
65 |
+
# For a 900 MW nuclear plant -> 500 000 MWh / month
|
66 |
+
# From https://www.edf.fr/groupe-edf/espaces-dedies/jeunes-enseignants/pour-les-jeunes/lenergie-de-a-a-z/produire-de-lelectricite/le-nucleaire-en-chiffres
|
67 |
+
YEARLY_NUCLEAR_ENERGY_EQ = q("6 TWh")
|
68 |
+
|
69 |
+
# From https://ourworldindata.org/population-growth
|
70 |
+
ONE_PERCENT_WORLD_POPULATION = 80_000_000
|
71 |
+
|
72 |
+
DAYS_IN_YEAR = 365
|
73 |
+
|
74 |
+
# For a 2MW wind turbine
|
75 |
+
# https://www.ecologie.gouv.fr/eolien-terrestre
|
76 |
+
YEARLY_WIND_ENERGY_EQ = q("4.2 GWh")
|
77 |
+
|
78 |
+
# Ireland yearly electricity consumption
|
79 |
+
# From https://en.wikipedia.org/wiki/List_of_countries_by_electricity_consumption
|
80 |
+
YEARLY_IRELAND_ELECTRICITY_CONSUMPTION = q("33 TWh")
|
81 |
+
IRELAND_POPULATION_MILLION = 5
|
82 |
+
|
83 |
+
# From https://impactco2.fr/outils/comparateur?value=1&comparisons=&equivalent=avion-pny
|
84 |
+
AIRPLANE_PARIS_NYC_GWP_EQ = q("1770 kgCO2eq")
|
85 |
+
|
86 |
+
|
87 |
+
def format_energy(energy: Energy) -> Quantity:
|
88 |
+
val = q(energy.value, energy.unit)
|
89 |
+
if val < q("1 kWh"):
|
90 |
+
val = val.to("Wh")
|
91 |
+
return val
|
92 |
+
|
93 |
+
|
94 |
+
def format_gwp(gwp: GWP) -> Quantity:
|
95 |
+
val = q(gwp.value, gwp.unit)
|
96 |
+
if val < q("1 kgCO2eq"):
|
97 |
+
val = val.to("gCO2eq")
|
98 |
+
return val
|
99 |
+
|
100 |
+
|
101 |
+
def format_adpe(adpe: ADPe) -> Quantity:
|
102 |
+
return q(adpe.value, adpe.unit)
|
103 |
+
|
104 |
+
|
105 |
+
def format_pe(pe: PE) -> Quantity:
|
106 |
+
val = q(pe.value, pe.unit)
|
107 |
+
if val < q("1 MJ"):
|
108 |
+
val = val.to("kJ")
|
109 |
+
return val
|
110 |
+
|
111 |
+
|
112 |
+
def format_impacts(impacts: Impacts) -> QImpacts:
|
113 |
+
return QImpacts(
|
114 |
+
energy=format_energy(impacts.energy),
|
115 |
+
gwp=format_gwp(impacts.gwp),
|
116 |
+
adpe=format_adpe(impacts.adpe),
|
117 |
+
pe=format_pe(impacts.pe),
|
118 |
+
)
|
119 |
+
|
120 |
+
|
121 |
+
def format_energy_eq_physical_activity(energy: Quantity) -> tuple[PhysicalActivity, Quantity]:
|
122 |
+
energy = energy.to("kJ")
|
123 |
+
running_eq = energy / RUNNING_ENERGY_EQ
|
124 |
+
if running_eq > q("1 km"):
|
125 |
+
return PhysicalActivity.RUNNING, running_eq
|
126 |
+
|
127 |
+
walking_eq = energy / WALKING_ENERGY_EQ
|
128 |
+
if walking_eq < q("1 km"):
|
129 |
+
walking_eq = walking_eq.to("m")
|
130 |
+
return PhysicalActivity.WALKING, walking_eq
|
131 |
+
|
132 |
+
|
133 |
+
def format_energy_eq_electric_vehicle(energy: Quantity) -> Quantity:
|
134 |
+
energy = energy.to("kWh")
|
135 |
+
ev_eq = energy / EV_ENERGY_EQ
|
136 |
+
if ev_eq < q("1 km"):
|
137 |
+
ev_eq = ev_eq.to("m")
|
138 |
+
return ev_eq
|
139 |
+
|
140 |
+
|
141 |
+
def format_gwp_eq_streaming(gwp: Quantity) -> Quantity:
|
142 |
+
gwp = gwp.to("kgCO2eq")
|
143 |
+
streaming_eq = gwp * STREAMING_GWP_EQ
|
144 |
+
if streaming_eq < q("1 h"):
|
145 |
+
streaming_eq = streaming_eq.to("min")
|
146 |
+
if streaming_eq < q("1 min"):
|
147 |
+
streaming_eq = streaming_eq.to("s")
|
148 |
+
return streaming_eq
|
149 |
+
|
150 |
+
|
151 |
+
def format_energy_eq_electricity_production(energy: Quantity) -> tuple[EnergyProduction, float]:
|
152 |
+
electricity_eq = energy * ONE_PERCENT_WORLD_POPULATION * DAYS_IN_YEAR
|
153 |
+
electricity_eq = electricity_eq.to("TWh")
|
154 |
+
if electricity_eq > YEARLY_NUCLEAR_ENERGY_EQ:
|
155 |
+
return EnergyProduction.NUCLEAR, electricity_eq / YEARLY_NUCLEAR_ENERGY_EQ
|
156 |
+
electricity_eq = electricity_eq.to("GWh")
|
157 |
+
return EnergyProduction.WIND, electricity_eq / YEARLY_WIND_ENERGY_EQ
|
158 |
+
|
159 |
+
|
160 |
+
def format_energy_eq_electricity_consumption_ireland(energy: Quantity) -> float:
|
161 |
+
electricity_eq = energy * ONE_PERCENT_WORLD_POPULATION * DAYS_IN_YEAR
|
162 |
+
electricity_eq = electricity_eq.to("TWh")
|
163 |
+
return electricity_eq / YEARLY_IRELAND_ELECTRICITY_CONSUMPTION
|
164 |
+
|
165 |
+
|
166 |
+
def format_gwp_eq_airplane_paris_nyc(gwp: Quantity) -> float:
|
167 |
+
gwp_eq = gwp * ONE_PERCENT_WORLD_POPULATION * DAYS_IN_YEAR
|
168 |
+
gwp_eq = gwp_eq.to("kgCO2eq")
|
169 |
+
return gwp_eq / AIRPLANE_PARIS_NYC_GWP_EQ
|
170 |
+
|
171 |
+
|
172 |
+
if __name__ == '__main__':
|
173 |
+
# energy = 5590e-9 # GWh
|
174 |
+
# energy = 3.58e-9 # GWh
|
175 |
+
# val = q("5.59 kWh") # gpt4
|
176 |
+
val = q("0.448 Wh")
|
177 |
+
val = val.to("MWh")
|
178 |
+
pop = 80_000_000
|
179 |
+
days = 365
|
180 |
+
|
181 |
+
tot = val * pop * days
|
182 |
+
print(tot)
|