added flask application and new scripts

This commit is contained in:
Angeliki 2022-01-14 09:13:16 -06:00
parent 96905ab730
commit bf438eb5be
1483 changed files with 278337 additions and 0 deletions

1
.gitignore vendored
View File

@ -1,2 +1,3 @@
pdf/* pdf/*
images/* images/*
audio/*

View File

@ -3048,3 +3048,81 @@
83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET / HTTP/1.1" 200 1682 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36" 83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET / HTTP/1.1" 200 1682 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36"
83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36" 83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36"
83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET /favicon.ico HTTP/1.1" 404 496 "http://wordmord-ur.la/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36" 83.163.232.95 - - [13/Jan/2022:06:18:19 -0600] "GET /favicon.ico HTTP/1.1" 404 496 "http://wordmord-ur.la/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36"
173.231.60.198 - - [13/Jan/2022:09:14:28 -0600] "GET /robots.txt HTTP/1.1" 404 460 "-" "Mozilla/5.0 (compatible; Adsbot/3.1; +https://seostar.co/robot/)"
173.231.60.198 - - [13/Jan/2022:09:14:33 -0600] "GET / HTTP/1.1" 200 1645 "-" "Mozilla/5.0 (compatible; Adsbot/3.1; +https://seostar.co/robot/)"
54.36.148.91 - - [13/Jan/2022:10:34:48 -0600] "GET / HTTP/1.1" 200 1626 "-" "Mozilla/5.0 (compatible; AhrefsBot/7.0; +http://ahrefs.com/robot/)"
195.154.123.117 - - [13/Jan/2022:10:34:48 -0600] "GET /style.css HTTP/1.1" 200 1075 "http://wordmord-ur.la/" "Mozilla/5.0 (compatible; AhrefsBot/7.0; +http://ahrefs.com/robot/)"
180.163.220.66 - - [13/Jan/2022:11:16:07 -0600] "GET / HTTP/1.1" 200 1682 "http://baidu.com/" "Mozilla/5.0 (Linux; U; Android 8.1.0; zh-CN; EML-AL00 Build/HUAWEIEML-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 baidu.sogo.uc.UCBrowser/11.9.4.974 UWS/2.13.1.48 Mobile Safari/537.36 AliApp(DingTalk/4.5.11) com.alibaba.android.rimet/10487439 Channel/227200 language/zh-CN"
180.163.220.66 - - [13/Jan/2022:11:16:08 -0600] "GET /style.css HTTP/1.1" 200 1074 "http://wordmord-ur.la/" "Mozilla/5.0 (Linux; U; Android 8.1.0; zh-CN; EML-AL00 Build/HUAWEIEML-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 baidu.sogo.uc.UCBrowser/11.9.4.974 UWS/2.13.1.48 Mobile Safari/537.36 AliApp(DingTalk/4.5.11) com.alibaba.android.rimet/10487439 Channel/227200 language/zh-CN"
180.163.220.4 - - [13/Jan/2022:11:16:52 -0600] "GET / HTTP/1.1" 200 1682 "http://baidu.com/" "Mozilla/5.0 (Linux; U; Android 8.1.0; zh-CN; EML-AL00 Build/HUAWEIEML-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 baidu.sogo.uc.UCBrowser/11.9.4.974 UWS/2.13.1.48 Mobile Safari/537.36 AliApp(DingTalk/4.5.11) com.alibaba.android.rimet/10487439 Channel/227200 language/zh-CN"
180.163.220.4 - - [13/Jan/2022:11:16:52 -0600] "GET /style.css HTTP/1.1" 200 1074 "http://wordmord-ur.la/" "Mozilla/5.0 (Linux; U; Android 8.1.0; zh-CN; EML-AL00 Build/HUAWEIEML-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 baidu.sogo.uc.UCBrowser/11.9.4.974 UWS/2.13.1.48 Mobile Safari/537.36 AliApp(DingTalk/4.5.11) com.alibaba.android.rimet/10487439 Channel/227200 language/zh-CN"
180.163.220.4 - - [13/Jan/2022:11:16:52 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (Linux; U; Android 8.1.0; zh-CN; EML-AL00 Build/HUAWEIEML-AL00) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/57.0.2987.108 baidu.sogo.uc.UCBrowser/11.9.4.974 UWS/2.13.1.48 Mobile Safari/537.36 AliApp(DingTalk/4.5.11) com.alibaba.android.rimet/10487439 Channel/227200 language/zh-CN"
83.163.232.95 - - [13/Jan/2022:12:56:00 -0600] "GET / HTTP/1.1" 200 1682 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:12:56:01 -0600] "GET /style.css HTTP/1.1" 200 1074 "http://wordmord-ur.la/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:12:56:02 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:12:56:02 -0600] "GET /favicon.ico HTTP/1.1" 404 496 "http://wordmord-ur.la/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:12:56:12 -0600] "GET /transformations.html HTTP/1.1" 200 1241 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
49.234.43.244 - - [13/Jan/2022:13:04:35 -0600] "GET / HTTP/1.1" 200 1645 "-" "Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/81.0.4044.138 Mobile Safari/537.36"
207.46.13.177 - - [13/Jan/2022:13:55:31 -0600] "GET /atom.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.146 - - [13/Jan/2022:14:39:21 -0600] "GET /sitemaps.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
83.163.232.95 - - [13/Jan/2022:14:49:29 -0600] "GET /styles HTTP/1.1" 301 587 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:29 -0600] "GET /styles/ HTTP/1.1" 200 699 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:29 -0600] "GET /icons/blank.gif HTTP/1.1" 200 433 "http://wordmord-ur.la/styles/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:29 -0600] "GET /icons/folder.gif HTTP/1.1" 200 510 "http://wordmord-ur.la/styles/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:29 -0600] "GET /icons/back.gif HTTP/1.1" 200 502 "http://wordmord-ur.la/styles/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:31 -0600] "GET /styles/fonts/ HTTP/1.1" 200 824 "http://wordmord-ur.la/styles/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:49:32 -0600] "GET /icons/odf6otf-20x22.png HTTP/1.1" 200 1369 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:50:08 -0600] "GET / HTTP/1.1" 200 1682 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:50:10 -0600] "GET /style.css HTTP/1.1" 200 1074 "http://wordmord-ur.la/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [13/Jan/2022:14:50:10 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /styles/fonts/ HTTP/1.1" 200 825 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /icons/blank.gif HTTP/1.1" 200 433 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /icons/back.gif HTTP/1.1" 200 501 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /icons/odf6otf-20x22.png HTTP/1.1" 200 1370 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /icons/folder.gif HTTP/1.1" 200 511 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:24 -0600] "GET /favicon.ico HTTP/1.1" 404 496 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:29 -0600] "GET /styles/fonts/ACNixie/ HTTP/1.1" 200 776 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:29 -0600] "GET /icons/unknown.gif HTTP/1.1" 200 530 "http://wordmord-ur.la/styles/fonts/ACNixie/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:32 -0600] "GET /styles/fonts/ACNixie/AC-Nixie.ttf HTTP/1.1" 200 258379 "http://wordmord-ur.la/styles/fonts/ACNixie/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
83.163.232.95 - - [13/Jan/2022:14:52:37 -0600] "GET /workshopA.html HTTP/1.1" 200 815 "http://wordmord-ur.la/" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
46.84.172.73 - - [13/Jan/2022:14:52:39 -0600] "GET /styles/fonts/ACSciFly/ HTTP/1.1" 200 775 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
83.163.232.95 - - [13/Jan/2022:14:52:40 -0600] "GET /styles/fonts/ACNixie/AC-Nixie.ttf HTTP/1.1" 200 258379 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
46.84.172.73 - - [13/Jan/2022:14:52:40 -0600] "GET /styles/fonts/ACSciFly/ACSciFly.ttf HTTP/1.1" 200 113647 "http://wordmord-ur.la/styles/fonts/ACSciFly/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:47 -0600] "GET /styles/fonts/Solide-Mirage-master/ HTTP/1.1" 200 865 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:47 -0600] "GET /icons/text.gif HTTP/1.1" 200 514 "http://wordmord-ur.la/styles/fonts/Solide-Mirage-master/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:48 -0600] "GET /styles/fonts/Solide-Mirage-master/fonts/ HTTP/1.1" 200 787 "http://wordmord-ur.la/styles/fonts/Solide-Mirage-master/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:50 -0600] "GET /styles/fonts/Solide-Mirage-master/fonts/webfonts/ HTTP/1.1" 200 767 "http://wordmord-ur.la/styles/fonts/Solide-Mirage-master/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:52 -0600] "GET /styles/fonts/Solide-Mirage-master/fonts/webfonts/Etroit/ HTTP/1.1" 200 786 "http://wordmord-ur.la/styles/fonts/Solide-Mirage-master/fonts/webfonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:52:54 -0600] "GET /styles/fonts/Solide-Mirage-master/fonts/webfonts/Etroit/Solide_Mirage-Etroit_web.ttf HTTP/1.1" 200 75626 "http://wordmord-ur.la/styles/fonts/Solide-Mirage-master/fonts/webfonts/Etroit/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:53:05 -0600] "GET /styles/fonts/ HTTP/1.1" 200 825 "-" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:53:06 -0600] "GET /styles/fonts/ACPoiret/ HTTP/1.1" 200 775 "http://wordmord-ur.la/styles/fonts/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:53:06 -0600] "GET /icons/unknown.gif HTTP/1.1" 200 530 "http://wordmord-ur.la/styles/fonts/ACPoiret/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
46.84.172.73 - - [13/Jan/2022:14:53:07 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71447 "http://wordmord-ur.la/styles/fonts/ACPoiret/" "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; rv:96.0) Gecko/20100101 Firefox/96.0"
40.77.167.71 - - [13/Jan/2022:15:03:45 -0600] "GET /sitemap.xml.gz HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
40.77.167.71 - - [13/Jan/2022:15:05:16 -0600] "GET /sitemap.txt HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
157.55.39.134 - - [13/Jan/2022:15:28:45 -0600] "GET /sitemap.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.177 - - [13/Jan/2022:15:29:21 -0600] "GET /sitemap_index.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
162.55.81.190 - - [13/Jan/2022:18:04:03 -0600] "GET / HTTP/1.1" 200 3052 "-" "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/86.0.4240.75 Safari/537.36"
54.36.148.218 - - [13/Jan/2022:23:48:17 -0600] "GET /robots.txt HTTP/1.1" 404 441 "-" "Mozilla/5.0 (compatible; AhrefsBot/7.0; +http://ahrefs.com/robot/)"
31.13.103.118 - - [14/Jan/2022:03:04:09 -0600] "GET / HTTP/1.1" 206 1692 "-" "facebookexternalhit/1.1 (+http://www.facebook.com/externalhit_uatext.php)"
83.163.232.95 - - [14/Jan/2022:06:10:17 -0600] "GET /style.css HTTP/1.1" 200 1075 "http://wordmord-ur.la/workshopA.html" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [14/Jan/2022:06:10:18 -0600] "GET /styles/fonts/ACPoiret/AC-Poiret.ttf HTTP/1.1" 200 71446 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [14/Jan/2022:06:10:18 -0600] "GET /styles/fonts/ACNixie/AC-Nixie.ttf HTTP/1.1" 200 258379 "http://wordmord-ur.la/style.css" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [14/Jan/2022:06:10:18 -0600] "GET /favicon.ico HTTP/1.1" 404 496 "http://wordmord-ur.la/workshopA.html" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [14/Jan/2022:06:42:10 -0600] "GET /transformations.html HTTP/1.1" 200 1241 "-" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
83.163.232.95 - - [14/Jan/2022:06:42:10 -0600] "GET /style.css HTTP/1.1" 200 1074 "http://wordmord-ur.la/transformations.html" "Mozilla/5.0 (X11; Ubuntu; Linux x86_64; rv:95.0) Gecko/20100101 Firefox/95.0"
157.55.39.23 - - [14/Jan/2022:06:53:13 -0600] "GET /robots.txt HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.146 - - [14/Jan/2022:06:53:26 -0600] "GET /sitemaps.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
157.55.39.143 - - [14/Jan/2022:07:03:32 -0600] "GET /atom.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
40.77.167.71 - - [14/Jan/2022:07:17:00 -0600] "GET /sitemap.xml.gz HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.177 - - [14/Jan/2022:07:19:13 -0600] "GET /sitemap_index.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
40.77.167.71 - - [14/Jan/2022:07:28:30 -0600] "GET /sitemap.txt HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
157.55.39.23 - - [14/Jan/2022:07:35:37 -0600] "GET /robots.txt HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.177 - - [14/Jan/2022:07:35:54 -0600] "GET /atom.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.146 - - [14/Jan/2022:07:46:21 -0600] "GET /sitemaps.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
40.77.167.71 - - [14/Jan/2022:08:11:05 -0600] "GET /sitemap.xml.gz HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
157.55.39.134 - - [14/Jan/2022:08:11:43 -0600] "GET /sitemap.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
40.77.167.71 - - [14/Jan/2022:08:12:25 -0600] "GET /sitemap.txt HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
157.55.39.134 - - [14/Jan/2022:08:36:26 -0600] "GET /sitemap.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.177 - - [14/Jan/2022:08:37:10 -0600] "GET /sitemap_index.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"
207.46.13.177 - - [14/Jan/2022:08:58:31 -0600] "GET /atom.xml HTTP/1.1" 404 497 "-" "Mozilla/5.0 (compatible; bingbot/2.0; +http://www.bing.com/bingbot.htm)"

View File

View File

@ -0,0 +1,142 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<title></title>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8" />
<meta name='ocr-system' content='tesseract 3.04.01' />
<meta name='ocr-capabilities' content='ocr_page ocr_carea ocr_par ocr_line ocrx_word'/>
</head>
<body>
<div class='ocr_page' id='page_1' title='image "images/anthropoktonia.png"; bbox 0 0 2480 3508; ppageno 0'>
<div class='ocr_carea' id='block_1_1' title="bbox 610 316 1872 479">
<p class='ocr_par' dir='ltr' id='par_1_1' title="bbox 610 316 1872 479">
<span class='ocr_line' id='line_1_1' title="bbox 610 316 1872 349; baseline 0 -1; x_size 39.351353; x_descenders 7.3513517; x_ascenders 9.9459457"><span class='ocrx_word' id='word_1_1' title='bbox 610 316 869 348; x_wconf 93' lang='ell' dir='ltr'>ΕΓΚΛΗΜΑΤΑ</span> <span class='ocrx_word' id='word_1_2' title='bbox 884 316 993 348; x_wconf 96' lang='ell' dir='ltr'>ΚΑΤΑ</span> <span class='ocrx_word' id='word_1_3' title='bbox 1005 316 1083 348; x_wconf 95' lang='ell' dir='ltr'>ΤΗΣ</span> <span class='ocrx_word' id='word_1_4' title='bbox 1097 316 1207 348; x_wconf 95' lang='ell' dir='ltr'>ΖΩΗΣ</span> <span class='ocrx_word' id='word_1_5' title='bbox 1223 316 1288 348; x_wconf 96' lang='ell' dir='ltr'>ΚΑΙ</span> <span class='ocrx_word' id='word_1_6' title='bbox 1305 316 1556 349; x_wconf 93' lang='ell' dir='ltr'>ΠΡΟΣΒΟΛΕΣ</span> <span class='ocrx_word' id='word_1_7' title='bbox 1568 316 1652 349; x_wconf 92' lang='ell' dir='ltr'>ΤΩΝ</span> <span class='ocrx_word' id='word_1_8' title='bbox 1667 316 1872 349; x_wconf 92' lang='ell' dir='ltr'>ΘΥΛΗΚΟΤΗΤ*</span>
</span>
<span class='ocr_line' id='line_1_2' title="bbox 769 435 1711 479; baseline 0.001 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_9' title='bbox 769 438 789 470; x_wconf 94' lang='ell' dir='ltr'>Ι.</span> <span class='ocrx_word' id='word_1_10' title='bbox 807 435 1033 479; x_wconf 89' lang='ell' dir='ltr'>Εγκλήματα</span> <span class='ocrx_word' id='word_1_11' title='bbox 1048 435 1200 479; x_wconf 90' lang='ell' dir='ltr'>βλάβης</span> <span class='ocrx_word' id='word_1_12' title='bbox 1212 446 1278 479; x_wconf 91' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_13' title='bbox 1293 435 1391 479; x_wconf 90' lang='ell' dir='ltr'>ζωής</span> <span class='ocrx_word' id='word_1_14' title='bbox 1404 446 1474 471; x_wconf 91' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_15' title='bbox 1491 435 1711 479; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_2' title="bbox 983 557 1496 662">
<p class='ocr_par' dir='ltr' id='par_1_2' title="bbox 983 557 1496 662">
<span class='ocr_line' id='line_1_3' title="bbox 1129 557 1351 601; baseline 0.005 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_16' title='bbox 1129 557 1264 601; x_wconf 92' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_17' title='bbox 1279 560 1351 593; x_wconf 89' lang='ell'>299</span>
</span>
<span class='ocr_line' id='line_1_4' title="bbox 983 618 1496 662; baseline 0.002 -9; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_18' title='bbox 983 618 1319 662; x_wconf 91' lang='ell' dir='ltr'>Γυναικοκτονία</span> <span class='ocrx_word' id='word_1_19' title='bbox 1334 629 1380 662; x_wconf 93' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_20' title='bbox 1395 618 1496 654; x_wconf 92' lang='ell' dir='ltr'>δόλο</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_3' title="bbox 243 740 2197 845">
<p class='ocr_par' dir='ltr' id='par_1_3' title="bbox 243 740 2197 845">
<span class='ocr_line' id='line_1_5' title="bbox 244 740 2166 784; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_21' title='bbox 244 743 274 775; x_wconf 93' lang='ell'>1.</span> <span class='ocrx_word' id='word_1_22' title='bbox 288 740 435 784; x_wconf 89' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_23' title='bbox 450 740 625 775; x_wconf 88' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_24' title='bbox 641 740 761 775; x_wconf 91' lang='ell' dir='ltr'>άλλη</span> <span class='ocrx_word' id='word_1_25' title='bbox 774 740 994 784; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_26' title='bbox 1009 751 1052 784; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_27' title='bbox 1070 740 1247 784; x_wconf 88' lang='ell' dir='ltr'>κάθειρξη</span> <span class='ocrx_word' id='word_1_28' title='bbox 1266 740 1395 784; x_wconf 89' lang='ell' dir='ltr'>ισόβια</span> <span class='ocrx_word' id='word_1_29' title='bbox 1412 740 1431 784; x_wconf 90' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_30' title='bbox 1448 740 1666 784; x_wconf 90' lang='ell' dir='ltr'>πρόσκαιρη</span> <span class='ocrx_word' id='word_1_31' title='bbox 1681 740 1933 784; x_wconf 84' lang='ell' dir='ltr'>τουλάχιστον</span> <span class='ocrx_word' id='word_1_32' title='bbox 1948 740 2045 775; x_wconf 90' lang='ell' dir='ltr'>δέκα</span> <span class='ocrx_word' id='word_1_33' title='bbox 2060 740 2166 775; x_wconf 93' lang='ell' dir='ltr'>ετών.</span>
</span>
<span class='ocr_line' id='line_1_6' title="bbox 243 801 2197 845; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_34' title='bbox 243 804 274 836; x_wconf 93' lang='ell'>2.</span> <span class='ocrx_word' id='word_1_35' title='bbox 291 804 340 836; x_wconf 93' lang='ell' dir='ltr'>Αν</span> <span class='ocrx_word' id='word_1_36' title='bbox 357 812 376 845; x_wconf 94' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_37' title='bbox 392 801 514 845; x_wconf 88' lang='ell' dir='ltr'>πράξη</span> <span class='ocrx_word' id='word_1_38' title='bbox 531 801 827 845; x_wconf 90' lang='ell' dir='ltr'>αποφασίστηκε</span> <span class='ocrx_word' id='word_1_39' title='bbox 846 812 907 836; x_wconf 90' lang='ell' dir='ltr'>και</span> <span class='ocrx_word' id='word_1_40' title='bbox 920 801 1166 845; x_wconf 92' lang='ell' dir='ltr'>εκτελέστηκε</span> <span class='ocrx_word' id='word_1_41' title='bbox 1183 812 1227 836; x_wconf 94' lang='ell' dir='ltr'>σε</span> <span class='ocrx_word' id='word_1_42' title='bbox 1245 801 1399 845; x_wconf 90' lang='ell' dir='ltr'>βρασμό</span> <span class='ocrx_word' id='word_1_43' title='bbox 1417 801 1579 845; x_wconf 88' lang='ell' dir='ltr'>ψυχικής</span> <span class='ocrx_word' id='word_1_44' title='bbox 1594 801 1728 845; x_wconf 91' lang='ell' dir='ltr'>ορμής,</span> <span class='ocrx_word' id='word_1_45' title='bbox 1744 801 1992 845; x_wconf 90' lang='ell' dir='ltr'>επιβάλλεται</span> <span class='ocrx_word' id='word_1_46' title='bbox 2007 801 2197 845; x_wconf 88' lang='ell' dir='ltr'>κάθειρξη.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_4' title="bbox 917 923 1561 1028">
<p class='ocr_par' dir='ltr' id='par_1_4' title="bbox 917 923 1561 1028">
<span class='ocr_line' id='line_1_7' title="bbox 1129 923 1352 967; baseline 0.004 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_47' title='bbox 1129 923 1264 967; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_48' title='bbox 1279 926 1352 959; x_wconf 91' lang='ell'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_8' title="bbox 917 984 1561 1028; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_49' title='bbox 917 984 1253 1028; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_50' title='bbox 1267 985 1348 1020; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_51' title='bbox 1364 984 1561 1028; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_5' title="bbox 238 1106 2235 1211">
<p class='ocr_par' dir='ltr' id='par_1_5' title="bbox 238 1106 2235 1211">
<span class='ocr_line' id='line_1_9' title="bbox 238 1106 2235 1150; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_52' title='bbox 238 1106 386 1150; x_wconf 88' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_53' title='bbox 407 1106 636 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_54' title='bbox 660 1117 722 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_55' title='bbox 742 1106 920 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_56' title='bbox 943 1106 1264 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_57' title='bbox 1286 1106 1432 1150; x_wconf 86' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_58' title='bbox 1453 1106 1531 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_59' title='bbox 1554 1106 1754 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_60' title='bbox 1777 1117 1838 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_61' title='bbox 1858 1106 2019 1150; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_62' title='bbox 2043 1106 2235 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_10' title="bbox 240 1167 2195 1211; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_63' title='bbox 240 1178 309 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_64' title='bbox 326 1167 499 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_65' title='bbox 515 1178 576 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_66' title='bbox 589 1167 667 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_67' title='bbox 683 1167 787 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_68' title='bbox 802 1168 845 1211; x_wconf 89' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_69' title='bbox 861 1167 981 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_70' title='bbox 995 1178 1071 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_71' title='bbox 1088 1167 1234 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_72' title='bbox 1250 1167 1328 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_73' title='bbox 1344 1167 1477 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_74' title='bbox 1494 1167 1682 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_75' title='bbox 1695 1167 1914 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_76' title='bbox 1929 1178 1972 1211; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_77' title='bbox 1988 1167 2195 1211; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_6' title="bbox 972 1289 1510 1394">
<p class='ocr_par' dir='ltr' id='par_1_6' title="bbox 972 1289 1510 1394">
<span class='ocr_line' id='line_1_11' title="bbox 1129 1289 1351 1333; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_78' title='bbox 1129 1289 1264 1333; x_wconf 94' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_79' title='bbox 1279 1292 1351 1325; x_wconf 91' lang='ell'>301</span>
</span>
<span class='ocr_line' id='line_1_12' title="bbox 972 1350 1510 1394; baseline 0 -8; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_80' title='bbox 972 1350 1193 1394; x_wconf 91' lang='ell' dir='ltr'>Συμμετοχή</span> <span class='ocrx_word' id='word_1_81' title='bbox 1210 1361 1256 1386; x_wconf 93' lang='ell' dir='ltr'>σε</span> <span class='ocrx_word' id='word_1_82' title='bbox 1271 1350 1510 1386; x_wconf 91' lang='ell' dir='ltr'>αυτοκτονία</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_7' title="bbox 238 1472 2238 1638">
<p class='ocr_par' dir='ltr' id='par_1_7' title="bbox 238 1472 2238 1638">
<span class='ocr_line' id='line_1_13' title="bbox 238 1472 2236 1516; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_83' title='bbox 238 1472 386 1516; x_wconf 92' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_84' title='bbox 412 1472 612 1507; x_wconf 90' lang='ell' dir='ltr'>κατέπεισε</span> <span class='ocrx_word' id='word_1_85' title='bbox 637 1472 758 1507; x_wconf 92' lang='ell' dir='ltr'>άλλην</span> <span class='ocrx_word' id='word_1_86' title='bbox 781 1483 829 1507; x_wconf 90' lang='ell' dir='ltr'>να</span> <span class='ocrx_word' id='word_1_87' title='bbox 853 1472 1142 1516; x_wconf 90' lang='ell' dir='ltr'>αυτοκτονήσει,</span> <span class='ocrx_word' id='word_1_88' title='bbox 1168 1483 1216 1507; x_wconf 90' lang='ell' dir='ltr'>αν</span> <span class='ocrx_word' id='word_1_89' title='bbox 1238 1472 1442 1516; x_wconf 92' lang='ell' dir='ltr'>τελέστηκε</span> <span class='ocrx_word' id='word_1_90' title='bbox 1469 1483 1488 1516; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_91' title='bbox 1514 1472 1745 1507; x_wconf 90' lang='ell' dir='ltr'>αυτοκτονία</span> <span class='ocrx_word' id='word_1_92' title='bbox 1771 1472 1790 1516; x_wconf 98' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_93' title='bbox 1816 1472 1915 1516; x_wconf 88' lang='ell' dir='ltr'>έγινε</span> <span class='ocrx_word' id='word_1_94' title='bbox 1941 1472 2138 1516; x_wconf 90' lang='ell' dir='ltr'>απόπειρά</span> <span class='ocrx_word' id='word_1_95' title='bbox 2160 1483 2236 1516; x_wconf 98' lang='ell' dir='ltr'>της,</span>
</span>
<span class='ocr_line' id='line_1_14' title="bbox 244 1533 2238 1577; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_96' title='bbox 244 1533 371 1577; x_wconf 90' lang='ell' dir='ltr'>καθώς</span> <span class='ocrx_word' id='word_1_97' title='bbox 390 1544 450 1568; x_wconf 90' lang='ell' dir='ltr'>και</span> <span class='ocrx_word' id='word_1_98' title='bbox 464 1533 602 1577; x_wconf 95' lang='ell' dir='ltr'>όποιος</span> <span class='ocrx_word' id='word_1_99' title='bbox 618 1533 746 1568; x_wconf 91' lang='ell' dir='ltr'>έδωσε</span> <span class='ocrx_word' id='word_1_100' title='bbox 766 1533 932 1577; x_wconf 90' lang='ell' dir='ltr'>βοήθεια</span> <span class='ocrx_word' id='word_1_101' title='bbox 950 1533 1043 1568; x_wconf 90' lang='ell' dir='ltr'>κατά</span> <span class='ocrx_word' id='word_1_102' title='bbox 1056 1544 1124 1577; x_wconf 96' lang='ell' dir='ltr'>την</span> <span class='ocrx_word' id='word_1_103' title='bbox 1138 1533 1275 1577; x_wconf 92' lang='ell' dir='ltr'>τέλεσή</span> <span class='ocrx_word' id='word_1_104' title='bbox 1291 1544 1367 1577; x_wconf 98' lang='ell' dir='ltr'>της,</span> <span class='ocrx_word' id='word_1_105' title='bbox 1387 1544 1406 1577; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_106' title='bbox 1424 1533 1543 1568; x_wconf 90' lang='ell' dir='ltr'>οποία</span> <span class='ocrx_word' id='word_1_107' title='bbox 1559 1533 1814 1577; x_wconf 90' lang='ell' dir='ltr'>διαφορετικά</span> <span class='ocrx_word' id='word_1_108' title='bbox 1830 1534 1900 1568; x_wconf 91' lang='ell' dir='ltr'>δεν</span> <span class='ocrx_word' id='word_1_109' title='bbox 1916 1534 1968 1568; x_wconf 90' lang='ell' dir='ltr'>θα</span> <span class='ocrx_word' id='word_1_110' title='bbox 1985 1533 2077 1577; x_wconf 90' lang='ell' dir='ltr'>ήταν</span> <span class='ocrx_word' id='word_1_111' title='bbox 2093 1533 2238 1577; x_wconf 97' lang='ell' dir='ltr'>εφικτή,</span>
</span>
<span class='ocr_line' id='line_1_15' title="bbox 240 1594 741 1638; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_112' title='bbox 240 1594 459 1638; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_113' title='bbox 474 1605 518 1638; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_114' title='bbox 534 1594 741 1638; x_wconf 92' lang='ell' dir='ltr'>φυλάκιση.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_8' title="bbox 933 1717 1548 1822">
<p class='ocr_par' dir='ltr' id='par_1_8' title="bbox 933 1717 1548 1822">
<span class='ocr_line' id='line_1_16' title="bbox 1129 1717 1351 1761; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_115' title='bbox 1129 1717 1264 1761; x_wconf 99' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_116' title='bbox 1279 1720 1351 1753; x_wconf 91' lang='ell'>302</span>
</span>
<span class='ocr_line' id='line_1_17' title="bbox 933 1778 1548 1822; baseline 0.002 -9; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_117' title='bbox 933 1778 1269 1822; x_wconf 91' lang='ell' dir='ltr'>Γυναικοκτονία</span> <span class='ocrx_word' id='word_1_118' title='bbox 1282 1778 1364 1814; x_wconf 99' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_119' title='bbox 1379 1778 1548 1822; x_wconf 91' lang='ell' dir='ltr'>θεσμική αμέλεια</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_9' title="bbox 238 1900 2047 1944">
<p class='ocr_par' dir='ltr' id='par_1_9' title="bbox 238 1900 2047 1944">
<span class='ocr_line' id='line_1_18' title="bbox 238 1900 2047 1944; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_120' title='bbox 238 1900 386 1944; x_wconf 95' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_121' title='bbox 401 1900 479 1935; x_wconf 90' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_122' title='bbox 495 1900 658 1944; x_wconf 90' lang='ell' dir='ltr'>αμέλεια</span> <span class='ocrx_word' id='word_1_123' title='bbox 673 1900 847 1935; x_wconf 94' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_124' title='bbox 863 1900 994 1942; x_wconf 92' lang='ell' dir='ltr'>άλλην,</span> <span class='ocrx_word' id='word_1_125' title='bbox 1009 1900 1227 1944; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_126' title='bbox 1242 1911 1285 1944; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_127' title='bbox 1302 1900 1496 1944; x_wconf 92' lang='ell' dir='ltr'>φυλάκιση</span> <span class='ocrx_word' id='word_1_128' title='bbox 1512 1900 1764 1944; x_wconf 92' lang='ell' dir='ltr'>τουλάχιστον</span> <span class='ocrx_word' id='word_1_129' title='bbox 1777 1900 1891 1944; x_wconf 93' lang='ell' dir='ltr'>τριών</span> <span class='ocrx_word' id='word_1_130' title='bbox 1908 1900 2047 1944; x_wconf 91' lang='ell' dir='ltr'>μηνών.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_10' title="bbox 1110 2022 1373 2119">
<p class='ocr_par' dir='ltr' id='par_1_10' title="bbox 1110 2022 1373 2119">
<span class='ocr_line' id='line_1_19' title="bbox 1129 2022 1351 2066; baseline 0.005 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_131' title='bbox 1129 2022 1264 2066; x_wconf 99' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_132' title='bbox 1279 2025 1351 2058; x_wconf 91' lang='ell'>303</span>
</span>
<span class='ocr_line' id='line_1_20' title="bbox 1110 2083 1373 2119; baseline 0 0; x_size 41.532211; x_descenders 5.5322127; x_ascenders 11"><span class='ocrx_word' id='word_1_133' title='bbox 1110 2083 1373 2119; x_wconf 91' lang='ell' dir='ltr'>Παιδοκτονία</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_11' title="bbox 242 2205 2238 2310">
<p class='ocr_par' dir='ltr' id='par_1_11' title="bbox 242 2205 2238 2310">
<span class='ocr_line' id='line_1_21' title="bbox 244 2205 2238 2249; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_134' title='bbox 244 2205 404 2249; x_wconf 89' lang='ell' dir='ltr'>Πατέρας</span> <span class='ocrx_word' id='word_1_135' title='bbox 418 2216 494 2240; x_wconf 95' lang='ell' dir='ltr'>που</span> <span class='ocrx_word' id='word_1_136' title='bbox 514 2216 557 2249; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_137' title='bbox 572 2205 749 2249; x_wconf 94' lang='ell' dir='ltr'>πρόθεση</span> <span class='ocrx_word' id='word_1_138' title='bbox 767 2205 941 2240; x_wconf 96' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_139' title='bbox 955 2216 998 2240; x_wconf 96' lang='ell' dir='ltr'>την</span> <span class='ocrx_word' id='word_1_140' title='bbox 1014 2205 1122 2240; x_wconf 90' lang='ell' dir='ltr'>κόρη</span> <span class='ocrx_word' id='word_1_141' title='bbox 1133 2216 1198 2249; x_wconf 98' lang='ell' dir='ltr'>του</span> <span class='ocrx_word' id='word_1_142' title='bbox 1215 2205 1309 2240; x_wconf 90' lang='ell' dir='ltr'>κατά</span> <span class='ocrx_word' id='word_1_143' title='bbox 1326 2205 1345 2249; x_wconf 98' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_144' title='bbox 1364 2205 1456 2249; x_wconf 91' lang='ell' dir='ltr'>μετά</span> <span class='ocrx_word' id='word_1_145' title='bbox 1469 2216 1536 2240; x_wconf 96' lang='ell' dir='ltr'>τον</span> <span class='ocrx_word' id='word_1_146' title='bbox 1548 2205 1694 2247; x_wconf 95' lang='ell' dir='ltr'>τοκετό,</span> <span class='ocrx_word' id='word_1_147' title='bbox 1711 2205 1810 2240; x_wconf 90' lang='ell' dir='ltr'>αλλά</span> <span class='ocrx_word' id='word_1_148' title='bbox 1825 2205 1901 2240; x_wconf 93' lang='ell' dir='ltr'>ενώ</span> <span class='ocrx_word' id='word_1_149' title='bbox 1917 2205 2238 2249; x_wconf 90' lang='ell' dir='ltr'>εξακολουθούσε</span>
</span>
<span class='ocr_line' id='line_1_22' title="bbox 242 2266 2077 2310; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_150' title='bbox 242 2266 368 2310; x_wconf 90' lang='ell' dir='ltr'>ακόμη</span> <span class='ocrx_word' id='word_1_151' title='bbox 387 2277 406 2310; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_152' title='bbox 424 2266 633 2310; x_wconf 90' lang='ell' dir='ltr'>διατάραξη</span> <span class='ocrx_word' id='word_1_153' title='bbox 648 2277 717 2301; x_wconf 95' lang='ell' dir='ltr'>του</span> <span class='ocrx_word' id='word_1_154' title='bbox 734 2266 974 2310; x_wconf 90' lang='ell' dir='ltr'>οργανισμού</span> <span class='ocrx_word' id='word_1_155' title='bbox 989 2277 1054 2310; x_wconf 98' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_156' title='bbox 1069 2266 1147 2301; x_wconf 90' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_157' title='bbox 1163 2266 1293 2308; x_wconf 90' lang='ell' dir='ltr'>αυτόν,</span> <span class='ocrx_word' id='word_1_158' title='bbox 1308 2266 1527 2310; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_159' title='bbox 1542 2277 1585 2310; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_160' title='bbox 1604 2266 1781 2310; x_wconf 93' lang='ell' dir='ltr'>κάθειρξη</span> <span class='ocrx_word' id='word_1_161' title='bbox 1799 2266 1874 2310; x_wconf 98' lang='ell' dir='ltr'>έως</span> <span class='ocrx_word' id='word_1_162' title='bbox 1889 2266 1986 2301; x_wconf 90' lang='ell' dir='ltr'>δέκα</span> <span class='ocrx_word' id='word_1_163' title='bbox 2000 2266 2077 2310; x_wconf 98' lang='ell' dir='ltr'>έτη.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_12' title="bbox 959 2449 1521 2493">
<p class='ocr_par' dir='ltr' id='par_1_12' title="bbox 959 2449 1521 2493">
<span class='ocr_line' id='line_1_23' title="bbox 959 2449 1521 2493; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_164' title='bbox 959 2452 993 2484; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_165' title='bbox 1011 2449 1240 2493; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_166' title='bbox 1253 2460 1323 2485; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_167' title='bbox 1340 2449 1521 2493; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_13' title="bbox 1030 2571 1450 2676">
<p class='ocr_par' dir='ltr' id='par_1_13' title="bbox 1030 2571 1450 2676">
<span class='ocr_line' id='line_1_24' title="bbox 1129 2571 1352 2615; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_168' title='bbox 1129 2571 1264 2615; x_wconf 99' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_169' title='bbox 1279 2574 1352 2607; x_wconf 91' lang='ell'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_25' title="bbox 1030 2632 1450 2676; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_170' title='bbox 1030 2632 1206 2676; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_171' title='bbox 1220 2643 1286 2676; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_172' title='bbox 1302 2632 1450 2676; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_14' title="bbox 240 2754 2240 3104">
<p class='ocr_par' dir='ltr' id='par_1_14' title="bbox 242 2754 2240 2859">
<span class='ocr_line' id='line_1_26' title="bbox 244 2754 2240 2798; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_173' title='bbox 244 2757 274 2789; x_wconf 93' lang='ell'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_174' title='bbox 292 2754 440 2798; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_175' title='bbox 457 2754 572 2798; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_176' title='bbox 589 2765 632 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_177' title='bbox 653 2754 864 2798; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_178' title='bbox 884 2765 949 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_179' title='bbox 968 2754 1111 2798; x_wconf 87' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_180' title='bbox 1133 2754 1332 2789; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_181' title='bbox 1347 2765 1415 2798; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_182' title='bbox 1435 2754 1558 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_183' title='bbox 1577 2765 1643 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_184' title='bbox 1660 2754 1879 2798; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_185' title='bbox 1899 2765 1942 2798; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_186' title='bbox 1965 2754 2142 2798; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_187' title='bbox 2166 2754 2240 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_27' title="bbox 242 2815 430 2859; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_188' title='bbox 242 2815 339 2850; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_189' title='bbox 353 2815 430 2859; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
<p class='ocr_par' dir='ltr' id='par_1_15' title="bbox 240 2876 2237 3104">
<span class='ocr_line' id='line_1_28' title="bbox 243 2876 2235 2920; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_190' title='bbox 243 2879 274 2911; x_wconf 93' lang='ell'>2.</span> <span class='ocrx_word' id='word_1_191' title='bbox 302 2876 449 2920; x_wconf 95' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_192' title='bbox 480 2887 524 2920; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_193' title='bbox 552 2887 595 2920; x_wconf 98' lang='ell' dir='ltr'>τη</span> <span class='ocrx_word' id='word_1_194' title='bbox 626 2876 837 2920; x_wconf 90' lang='ell' dir='ltr'>συναίνεση</span> <span class='ocrx_word' id='word_1_195' title='bbox 866 2887 931 2920; x_wconf 98' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_196' title='bbox 960 2876 1102 2920; x_wconf 88' lang='ell' dir='ltr'>εγκύου</span> <span class='ocrx_word' id='word_1_197' title='bbox 1136 2876 1155 2920; x_wconf 98' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_198' title='bbox 1184 2888 1259 2911; x_wconf 96' lang='ell' dir='ltr'>των</span> <span class='ocrx_word' id='word_1_199' title='bbox 1287 2876 1510 2920; x_wconf 93' lang='ell' dir='ltr'>προσώπων</span> <span class='ocrx_word' id='word_1_200' title='bbox 1538 2887 1614 2911; x_wconf 95' lang='ell' dir='ltr'>που</span> <span class='ocrx_word' id='word_1_201' title='bbox 1645 2876 1761 2920; x_wconf 95' lang='ell' dir='ltr'>έχουν</span> <span class='ocrx_word' id='word_1_202' title='bbox 1788 2887 1831 2920; x_wconf 98' lang='ell' dir='ltr'>τη</span> <span class='ocrx_word' id='word_1_203' title='bbox 1861 2876 1990 2920; x_wconf 95' lang='ell' dir='ltr'>γονική</span> <span class='ocrx_word' id='word_1_204' title='bbox 2024 2876 2186 2920; x_wconf 90' lang='ell' dir='ltr'>μέριμνα</span> <span class='ocrx_word' id='word_1_205' title='bbox 2216 2876 2235 2920; x_wconf 98' lang='ell' dir='ltr'>ή</span>
</span>
<span class='ocr_line' id='line_1_29' title="bbox 242 2937 2236 2981; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_206' title='bbox 242 2937 441 2981; x_wconf 91' lang='ell' dir='ltr'>επιμέλειά</span> <span class='ocrx_word' id='word_1_207' title='bbox 454 2948 520 2981; x_wconf 98' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_208' title='bbox 535 2948 582 2972; x_wconf 90' lang='ell' dir='ltr'>αν</span> <span class='ocrx_word' id='word_1_209' title='bbox 598 2937 693 2981; x_wconf 90' lang='ell' dir='ltr'>αυτή</span> <span class='ocrx_word' id='word_1_210' title='bbox 711 2937 810 2972; x_wconf 90' lang='ell' dir='ltr'>είναι</span> <span class='ocrx_word' id='word_1_211' title='bbox 823 2937 982 2981; x_wconf 90' lang='ell' dir='ltr'>ανίκανη</span> <span class='ocrx_word' id='word_1_212' title='bbox 999 2948 1048 2972; x_wconf 90' lang='ell' dir='ltr'>να</span> <span class='ocrx_word' id='word_1_213' title='bbox 1063 2937 1296 2979; x_wconf 90' lang='ell' dir='ltr'>συναινέσει,</span> <span class='ocrx_word' id='word_1_214' title='bbox 1313 2937 1512 2972; x_wconf 90' lang='ell' dir='ltr'>διακόπτει</span> <span class='ocrx_word' id='word_1_215' title='bbox 1523 2948 1591 2981; x_wconf 96' lang='ell' dir='ltr'>την</span> <span class='ocrx_word' id='word_1_216' title='bbox 1606 2948 1695 2981; x_wconf 95' lang='ell' dir='ltr'>εγκυ</span> <span class='ocrx_word' id='word_1_217' title='bbox 1703 2937 1851 2981; x_wconf 92' lang='ell' dir='ltr'>μοσύνη</span> <span class='ocrx_word' id='word_1_218' title='bbox 1867 2948 1943 2981; x_wconf 98' lang='ell' dir='ltr'>της,</span> <span class='ocrx_word' id='word_1_219' title='bbox 1958 2937 2177 2981; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_220' title='bbox 2193 2948 2236 2981; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_30' title="bbox 242 2998 2237 3042; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_221' title='bbox 242 2998 437 3042; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_222' title='bbox 471 2998 546 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_223' title='bbox 576 2998 661 3042; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_224' title='bbox 693 2998 756 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_225' title='bbox 793 2998 812 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;</span> <span class='ocrx_word' id='word_1_226' title='bbox 844 2998 1052 3042; x_wconf 90' lang='ell' dir='ltr'>Σας</span> <span class='ocrx_word' id='word_1_227' title='bbox 1085 2998 1197 3042; x_wconf 95' lang='ell' dir='ltr'>καλούμε</span> <span class='ocrx_word' id='word_1_228' title='bbox 1233 3009 1294 3033; x_wconf 90' lang='ell' dir='ltr'>να</span> <span class='ocrx_word' id='word_1_229' title='bbox 1324 3009 1372 3033; x_wconf 90' lang='ell' dir='ltr'>συμμετέχετε</span> <span class='ocrx_word' id='word_1_230' title='bbox 1404 2998 1554 3042; x_wconf 96' lang='ell' dir='ltr'>σε</span> <span class='ocrx_word' id='word_1_231' title='bbox 1586 2999 1662 3033; x_wconf 90' lang='ell' dir='ltr'>μία</span> <span class='ocrx_word' id='word_1_232' title='bbox 1696 2998 1930 3042; x_wconf 90' lang='ell' dir='ltr'>συζήτηση</span> <span class='ocrx_word' id='word_1_233' title='bbox 1966 3009 2009 3042; x_wconf 91' lang='ell' dir='ltr'>διερώτησης</span> <span class='ocrx_word' id='word_1_234' title='bbox 2042 2998 2237 3042; x_wconf 92' lang='ell' dir='ltr'>του</span>
</span> <span class='ocr_line' id='line_1_31' title="bbox 240 3060 2235 3104; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_235' title='bbox 240 3060 493 3104; x_wconf 92' lang='ell' dir='ltr'>νόμου</span> <span class='ocrx_word' id='word_1_236' title='bbox 516 3060 591 3095; x_wconf 91' lang='ell' dir='ltr'>μέσω</span> <span class='ocrx_word' id='word_1_237' title='bbox 615 3060 712 3095; x_wconf 93' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_238' title='bbox 736 3071 797 3095; x_wconf 90' lang='ell' dir='ltr'>mailing</span> <span class='ocrx_word' id='word_1_239' title='bbox 817 3060 1025 3104; x_wconf 90' lang='ell' dir='ltr'>list:</span> <span class='ocrx_word' id='word_1_240' title='bbox 1050 3060 1175 3104; x_wconf 95' lang='ell' dir='ltr'>https://we.lurk.org/mailman3/lists/wordmord.we.lurk.org/</span>
<span class='ocrx_word' id='word_1_241' title='bbox 1202 3063 1261 3095; x_wconf 89' lang='ell' dir='ltr'>καμία*</span> <span class='ocrx_word' id='word_1_242' title='bbox 1284 3071 1351 3104; x_wconf 96' lang='ell' dir='ltr'>επισημείωση</span> <span class='ocrx_word' id='word_1_243' title='bbox 1376 3060 1452 3095; x_wconf 90' lang='ell' dir='ltr'>είναι</span> <span class='ocrx_word' id='word_1_244' title='bbox 1474 3060 1599 3104; x_wconf 95' lang='ell' dir='ltr'>μόνη*,</span> <span class='ocrx_word' id='word_1_245' title='bbox 1627 3060 1817 3104; x_wconf 92' lang='ell' dir='ltr'>WordMord,</span> <span class='ocrx_word' id='word_1_246' title='bbox 1845 3060 1938 3095; x_wconf 90' lang='ell' dir='ltr'>2022,</span> <span class='ocrx_word' id='word_1_247' title='bbox 1959 3071 2003 3095; x_wconf 95' lang='ell' dir='ltr'>Free</span> <span class='ocrx_word' id='word_1_248' title='bbox 2029 3060 2144 3104; x_wconf 91' lang='ell' dir='ltr'>Art</span> <span class='ocrx_word' id='word_1_249' title='bbox 2166 3071 2235 3095; x_wconf 95' lang='ell' dir='ltr'>License</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_15' title="bbox 2193 3257 2238 3292">
<p class='ocr_par' dir='ltr' id='par_1_16' title="bbox 2193 3257 2238 3292">
<span class='ocr_line' id='line_1_32' title="bbox 2193 3257 2238 3292; baseline 0 0; x_size 46.666668; x_descenders 11.666667; x_ascenders 11.666667"><span class='ocrx_word' id='word_1_250' title='bbox 2193 3257 2238 3292; x_wconf 88' lang='ell'>78</span>
</span>
</p>
</div>
</div>
</body>
</html>

View File

@ -0,0 +1,142 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<title></title>
<meta http-equiv="Content-Type" content="text/html;charset=utf-8" />
<meta name='ocr-system' content='tesseract 3.04.01' />
<meta name='ocr-capabilities' content='ocr_page ocr_carea ocr_par ocr_line ocrx_word'/>
</head>
<body>
<div class='ocr_page' id='page_1' title='image "images/anthropoktonia.png"; bbox 0 0 2480 3508; ppageno 0'>
<div class='ocr_carea' id='block_1_1' title="bbox 610 316 1872 479">
<p class='ocr_par' dir='ltr' id='par_1_1' title="bbox 610 316 1872 479">
<span class='ocr_line' id='line_1_1' title="bbox 610 316 1872 349; baseline 0 -1; x_size 39.351353; x_descenders 7.3513517; x_ascenders 9.9459457"><span class='ocrx_word' id='word_1_1' title='bbox 610 316 869 348; x_wconf 93' lang='ell' dir='ltr'>ΕΓΚΛΗΜΑΤΑ</span> <span class='ocrx_word' id='word_1_2' title='bbox 884 316 993 348; x_wconf 96' lang='ell' dir='ltr'>ΚΑΤΑ</span> <span class='ocrx_word' id='word_1_3' title='bbox 1005 316 1083 348; x_wconf 95' lang='ell' dir='ltr'>ΤΗΣ</span> <span class='ocrx_word' id='word_1_4' title='bbox 1097 316 1207 348; x_wconf 95' lang='ell' dir='ltr'>ΖΩΗΣ</span> <span class='ocrx_word' id='word_1_5' title='bbox 1223 316 1288 348; x_wconf 96' lang='ell' dir='ltr'>ΚΑΙ</span> <span class='ocrx_word' id='word_1_6' title='bbox 1305 316 1556 349; x_wconf 93' lang='ell' dir='ltr'>ΠΡΟΣΒΟΛΕΣ</span> <span class='ocrx_word' id='word_1_7' title='bbox 1568 316 1652 349; x_wconf 92' lang='ell' dir='ltr'>ΤΩΝ</span> <span class='ocrx_word' id='word_1_8' title='bbox 1667 316 1872 349; x_wconf 92' lang='ell' dir='ltr'>ΘΥΛΗΚΟΤΗΤ*</span>
</span>
<span class='ocr_line' id='line_1_2' title="bbox 769 435 1711 479; baseline 0.001 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_9' title='bbox 769 438 789 470; x_wconf 94' lang='ell' dir='ltr'>Ι.</span> <span class='ocrx_word' id='word_1_10' title='bbox 807 435 1033 479; x_wconf 89' lang='ell' dir='ltr'>Εγκλήματα</span> <span class='ocrx_word' id='word_1_11' title='bbox 1048 435 1200 479; x_wconf 90' lang='ell' dir='ltr'>βλάβης</span> <span class='ocrx_word' id='word_1_12' title='bbox 1212 446 1278 479; x_wconf 91' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_13' title='bbox 1293 435 1391 479; x_wconf 90' lang='ell' dir='ltr'>ζωής</span> <span class='ocrx_word' id='word_1_14' title='bbox 1404 446 1474 471; x_wconf 91' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_15' title='bbox 1491 435 1711 479; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_2' title="bbox 983 557 1496 662">
<p class='ocr_par' dir='ltr' id='par_1_2' title="bbox 983 557 1496 662">
<span class='ocr_line' id='line_1_3' title="bbox 1129 557 1351 601; baseline 0.005 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_16' title='bbox 1129 557 1264 601; x_wconf 92' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_17' title='bbox 1279 560 1351 593; x_wconf 89' lang='ell'>299</span>
</span>
<span class='ocr_line' id='line_1_4' title="bbox 983 618 1496 662; baseline 0.002 -9; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_18' title='bbox 983 618 1319 662; x_wconf 91' lang='ell' dir='ltr'>Γυναικοκτονία</span> <span class='ocrx_word' id='word_1_19' title='bbox 1334 629 1380 662; x_wconf 93' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_20' title='bbox 1395 618 1496 654; x_wconf 92' lang='ell' dir='ltr'>δόλο</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_3' title="bbox 243 740 2197 845">
<p class='ocr_par' dir='ltr' id='par_1_3' title="bbox 243 740 2197 845">
<span class='ocr_line' id='line_1_5' title="bbox 244 740 2166 784; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_21' title='bbox 244 743 274 775; x_wconf 93' lang='ell'>1.</span> <span class='ocrx_word' id='word_1_22' title='bbox 288 740 435 784; x_wconf 89' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_23' title='bbox 450 740 625 775; x_wconf 88' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_24' title='bbox 641 740 761 775; x_wconf 91' lang='ell' dir='ltr'>άλλη</span> <span class='ocrx_word' id='word_1_25' title='bbox 774 740 994 784; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_26' title='bbox 1009 751 1052 784; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_27' title='bbox 1070 740 1247 784; x_wconf 88' lang='ell' dir='ltr'>κάθειρξη</span> <span class='ocrx_word' id='word_1_28' title='bbox 1266 740 1395 784; x_wconf 89' lang='ell' dir='ltr'>ισόβια</span> <span class='ocrx_word' id='word_1_29' title='bbox 1412 740 1431 784; x_wconf 90' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_30' title='bbox 1448 740 1666 784; x_wconf 90' lang='ell' dir='ltr'>πρόσκαιρη</span> <span class='ocrx_word' id='word_1_31' title='bbox 1681 740 1933 784; x_wconf 84' lang='ell' dir='ltr'>τουλάχιστον</span> <span class='ocrx_word' id='word_1_32' title='bbox 1948 740 2045 775; x_wconf 90' lang='ell' dir='ltr'>δέκα</span> <span class='ocrx_word' id='word_1_33' title='bbox 2060 740 2166 775; x_wconf 93' lang='ell' dir='ltr'>ετών.</span>
</span>
<span class='ocr_line' id='line_1_6' title="bbox 243 801 2197 845; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_34' title='bbox 243 804 274 836; x_wconf 93' lang='ell'>2.</span> <span class='ocrx_word' id='word_1_35' title='bbox 291 804 340 836; x_wconf 93' lang='ell' dir='ltr'>Αν</span> <span class='ocrx_word' id='word_1_36' title='bbox 357 812 376 845; x_wconf 94' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_37' title='bbox 392 801 514 845; x_wconf 88' lang='ell' dir='ltr'>πράξη</span> <span class='ocrx_word' id='word_1_38' title='bbox 531 801 827 845; x_wconf 90' lang='ell' dir='ltr'>αποφασίστηκε</span> <span class='ocrx_word' id='word_1_39' title='bbox 846 812 907 836; x_wconf 90' lang='ell' dir='ltr'>και</span> <span class='ocrx_word' id='word_1_40' title='bbox 920 801 1166 845; x_wconf 92' lang='ell' dir='ltr'>εκτελέστηκε</span> <span class='ocrx_word' id='word_1_41' title='bbox 1183 812 1227 836; x_wconf 94' lang='ell' dir='ltr'>σε</span> <span class='ocrx_word' id='word_1_42' title='bbox 1245 801 1399 845; x_wconf 90' lang='ell' dir='ltr'>βρασμό</span> <span class='ocrx_word' id='word_1_43' title='bbox 1417 801 1579 845; x_wconf 88' lang='ell' dir='ltr'>ψυχικής</span> <span class='ocrx_word' id='word_1_44' title='bbox 1594 801 1728 845; x_wconf 91' lang='ell' dir='ltr'>ορμής,</span> <span class='ocrx_word' id='word_1_45' title='bbox 1744 801 1992 845; x_wconf 90' lang='ell' dir='ltr'>επιβάλλεται</span> <span class='ocrx_word' id='word_1_46' title='bbox 2007 801 2197 845; x_wconf 88' lang='ell' dir='ltr'>κάθειρξη.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_4' title="bbox 917 923 1561 1028">
<p class='ocr_par' dir='ltr' id='par_1_4' title="bbox 917 923 1561 1028">
<span class='ocr_line' id='line_1_7' title="bbox 1129 923 1352 967; baseline 0.004 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_47' title='bbox 1129 923 1264 967; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_48' title='bbox 1279 926 1352 959; x_wconf 91' lang='ell'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_8' title="bbox 917 984 1561 1028; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_49' title='bbox 917 984 1253 1028; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_50' title='bbox 1267 985 1348 1020; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_51' title='bbox 1364 984 1561 1028; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_5' title="bbox 238 1106 2235 1211">
<p class='ocr_par' dir='ltr' id='par_1_5' title="bbox 238 1106 2235 1211">
<span class='ocr_line' id='line_1_9' title="bbox 238 1106 2235 1150; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_52' title='bbox 238 1106 386 1150; x_wconf 88' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_53' title='bbox 407 1106 636 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_54' title='bbox 660 1117 722 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_55' title='bbox 742 1106 920 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_56' title='bbox 943 1106 1264 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_57' title='bbox 1286 1106 1432 1150; x_wconf 86' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_58' title='bbox 1453 1106 1531 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_59' title='bbox 1554 1106 1754 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_60' title='bbox 1777 1117 1838 1141; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_61' title='bbox 1858 1106 2019 1150; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_62' title='bbox 2043 1106 2235 1150; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_10' title="bbox 240 1167 2195 1211; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_63' title='bbox 240 1178 309 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_64' title='bbox 326 1167 499 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_65' title='bbox 515 1178 576 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_66' title='bbox 589 1167 667 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_67' title='bbox 683 1167 787 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_68' title='bbox 802 1168 845 1211; x_wconf 89' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_69' title='bbox 861 1167 981 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_70' title='bbox 995 1178 1071 1202; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_71' title='bbox 1088 1167 1234 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_72' title='bbox 1250 1167 1328 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_73' title='bbox 1344 1167 1477 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_74' title='bbox 1494 1167 1682 1202; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_75' title='bbox 1695 1167 1914 1211; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_76' title='bbox 1929 1178 1972 1211; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_77' title='bbox 1988 1167 2195 1211; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_6' title="bbox 972 1289 1510 1394">
<p class='ocr_par' dir='ltr' id='par_1_6' title="bbox 972 1289 1510 1394">
<span class='ocr_line' id='line_1_11' title="bbox 1129 1289 1351 1333; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_78' title='bbox 1129 1289 1264 1333; x_wconf 94' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_79' title='bbox 1279 1292 1351 1325; x_wconf 91' lang='ell'>301</span>
</span>
<span class='ocr_line' id='line_1_12' title="bbox 972 1350 1510 1394; baseline 0 -8; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_80' title='bbox 972 1350 1193 1394; x_wconf 91' lang='ell' dir='ltr'>Συμμετοχή</span> <span class='ocrx_word' id='word_1_81' title='bbox 1210 1361 1256 1386; x_wconf 93' lang='ell' dir='ltr'>σε</span> <span class='ocrx_word' id='word_1_82' title='bbox 1271 1350 1510 1386; x_wconf 91' lang='ell' dir='ltr'>αυτοκτονία</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_7' title="bbox 238 1472 2238 1638">
<p class='ocr_par' dir='ltr' id='par_1_7' title="bbox 238 1472 2238 1638">
<span class='ocr_line' id='line_1_13' title="bbox 238 1472 2236 1516; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_83' title='bbox 238 1472 386 1516; x_wconf 92' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_84' title='bbox 412 1472 612 1507; x_wconf 90' lang='ell' dir='ltr'>κατέπεισε</span> <span class='ocrx_word' id='word_1_85' title='bbox 637 1472 758 1507; x_wconf 92' lang='ell' dir='ltr'>άλλην</span> <span class='ocrx_word' id='word_1_86' title='bbox 781 1483 829 1507; x_wconf 90' lang='ell' dir='ltr'>να</span> <span class='ocrx_word' id='word_1_87' title='bbox 853 1472 1142 1516; x_wconf 90' lang='ell' dir='ltr'>αυτοκτονήσει,</span> <span class='ocrx_word' id='word_1_88' title='bbox 1168 1483 1216 1507; x_wconf 90' lang='ell' dir='ltr'>αν</span> <span class='ocrx_word' id='word_1_89' title='bbox 1238 1472 1442 1516; x_wconf 92' lang='ell' dir='ltr'>τελέστηκε</span> <span class='ocrx_word' id='word_1_90' title='bbox 1469 1483 1488 1516; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_91' title='bbox 1514 1472 1745 1507; x_wconf 90' lang='ell' dir='ltr'>αυτοκτονία</span> <span class='ocrx_word' id='word_1_92' title='bbox 1771 1472 1790 1516; x_wconf 98' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_93' title='bbox 1816 1472 1915 1516; x_wconf 88' lang='ell' dir='ltr'>έγινε</span> <span class='ocrx_word' id='word_1_94' title='bbox 1941 1472 2138 1516; x_wconf 90' lang='ell' dir='ltr'>απόπειρά</span> <span class='ocrx_word' id='word_1_95' title='bbox 2160 1483 2236 1516; x_wconf 98' lang='ell' dir='ltr'>της,</span>
</span>
<span class='ocr_line' id='line_1_14' title="bbox 244 1533 2238 1577; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_96' title='bbox 244 1533 371 1577; x_wconf 90' lang='ell' dir='ltr'>καθώς</span> <span class='ocrx_word' id='word_1_97' title='bbox 390 1544 450 1568; x_wconf 90' lang='ell' dir='ltr'>και</span> <span class='ocrx_word' id='word_1_98' title='bbox 464 1533 602 1577; x_wconf 95' lang='ell' dir='ltr'>όποιος</span> <span class='ocrx_word' id='word_1_99' title='bbox 618 1533 746 1568; x_wconf 91' lang='ell' dir='ltr'>έδωσε</span> <span class='ocrx_word' id='word_1_100' title='bbox 766 1533 932 1577; x_wconf 90' lang='ell' dir='ltr'>βοήθεια</span> <span class='ocrx_word' id='word_1_101' title='bbox 950 1533 1043 1568; x_wconf 90' lang='ell' dir='ltr'>κατά</span> <span class='ocrx_word' id='word_1_102' title='bbox 1056 1544 1124 1577; x_wconf 96' lang='ell' dir='ltr'>την</span> <span class='ocrx_word' id='word_1_103' title='bbox 1138 1533 1275 1577; x_wconf 92' lang='ell' dir='ltr'>τέλεσή</span> <span class='ocrx_word' id='word_1_104' title='bbox 1291 1544 1367 1577; x_wconf 98' lang='ell' dir='ltr'>της,</span> <span class='ocrx_word' id='word_1_105' title='bbox 1387 1544 1406 1577; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_106' title='bbox 1424 1533 1543 1568; x_wconf 90' lang='ell' dir='ltr'>οποία</span> <span class='ocrx_word' id='word_1_107' title='bbox 1559 1533 1814 1577; x_wconf 90' lang='ell' dir='ltr'>διαφορετικά</span> <span class='ocrx_word' id='word_1_108' title='bbox 1830 1534 1900 1568; x_wconf 91' lang='ell' dir='ltr'>δεν</span> <span class='ocrx_word' id='word_1_109' title='bbox 1916 1534 1968 1568; x_wconf 90' lang='ell' dir='ltr'>θα</span> <span class='ocrx_word' id='word_1_110' title='bbox 1985 1533 2077 1577; x_wconf 90' lang='ell' dir='ltr'>ήταν</span> <span class='ocrx_word' id='word_1_111' title='bbox 2093 1533 2238 1577; x_wconf 97' lang='ell' dir='ltr'>εφικτή,</span>
</span>
<span class='ocr_line' id='line_1_15' title="bbox 240 1594 741 1638; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_112' title='bbox 240 1594 459 1638; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_113' title='bbox 474 1605 518 1638; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_114' title='bbox 534 1594 741 1638; x_wconf 92' lang='ell' dir='ltr'>φυλάκιση.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_8' title="bbox 933 1717 1548 1822">
<p class='ocr_par' dir='ltr' id='par_1_8' title="bbox 933 1717 1548 1822">
<span class='ocr_line' id='line_1_16' title="bbox 1129 1717 1351 1761; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_115' title='bbox 1129 1717 1264 1761; x_wconf 99' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_116' title='bbox 1279 1720 1351 1753; x_wconf 91' lang='ell'>302</span>
</span>
<span class='ocr_line' id='line_1_17' title="bbox 933 1778 1548 1822; baseline 0.002 -9; x_size 45; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_117' title='bbox 933 1778 1269 1822; x_wconf 91' lang='ell' dir='ltr'>Γυναικοκτονία</span> <span class='ocrx_word' id='word_1_118' title='bbox 1282 1778 1364 1814; x_wconf 99' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_119' title='bbox 1379 1778 1548 1822; x_wconf 91' lang='ell' dir='ltr'>θεσμική αμέλεια</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_9' title="bbox 238 1900 2047 1944">
<p class='ocr_par' dir='ltr' id='par_1_9' title="bbox 238 1900 2047 1944">
<span class='ocr_line' id='line_1_18' title="bbox 238 1900 2047 1944; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_120' title='bbox 238 1900 386 1944; x_wconf 95' lang='ell' dir='ltr'>Όποιος</span> <span class='ocrx_word' id='word_1_121' title='bbox 401 1900 479 1935; x_wconf 90' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_122' title='bbox 495 1900 658 1944; x_wconf 90' lang='ell' dir='ltr'>αμέλεια</span> <span class='ocrx_word' id='word_1_123' title='bbox 673 1900 847 1935; x_wconf 94' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_124' title='bbox 863 1900 994 1942; x_wconf 92' lang='ell' dir='ltr'>άλλην,</span> <span class='ocrx_word' id='word_1_125' title='bbox 1009 1900 1227 1944; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_126' title='bbox 1242 1911 1285 1944; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_127' title='bbox 1302 1900 1496 1944; x_wconf 92' lang='ell' dir='ltr'>φυλάκιση</span> <span class='ocrx_word' id='word_1_128' title='bbox 1512 1900 1764 1944; x_wconf 92' lang='ell' dir='ltr'>τουλάχιστον</span> <span class='ocrx_word' id='word_1_129' title='bbox 1777 1900 1891 1944; x_wconf 93' lang='ell' dir='ltr'>τριών</span> <span class='ocrx_word' id='word_1_130' title='bbox 1908 1900 2047 1944; x_wconf 91' lang='ell' dir='ltr'>μηνών.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_10' title="bbox 1110 2022 1373 2119">
<p class='ocr_par' dir='ltr' id='par_1_10' title="bbox 1110 2022 1373 2119">
<span class='ocr_line' id='line_1_19' title="bbox 1129 2022 1351 2066; baseline 0.005 -9; x_size 43; x_descenders 8; x_ascenders 10"><span class='ocrx_word' id='word_1_131' title='bbox 1129 2022 1264 2066; x_wconf 99' lang='ell' dir='ltr'>Άρθρο</span> <span class='ocrx_word' id='word_1_132' title='bbox 1279 2025 1351 2058; x_wconf 91' lang='ell'>303</span>
</span>
<span class='ocr_line' id='line_1_20' title="bbox 1110 2083 1373 2119; baseline 0 0; x_size 41.532211; x_descenders 5.5322127; x_ascenders 11"><span class='ocrx_word' id='word_1_133' title='bbox 1110 2083 1373 2119; x_wconf 91' lang='ell' dir='ltr'>Παιδοκτονία</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_11' title="bbox 242 2205 2238 2310">
<p class='ocr_par' dir='ltr' id='par_1_11' title="bbox 242 2205 2238 2310">
<span class='ocr_line' id='line_1_21' title="bbox 244 2205 2238 2249; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_134' title='bbox 244 2205 404 2249; x_wconf 89' lang='ell' dir='ltr'>Πατέρας</span> <span class='ocrx_word' id='word_1_135' title='bbox 418 2216 494 2240; x_wconf 95' lang='ell' dir='ltr'>που</span> <span class='ocrx_word' id='word_1_136' title='bbox 514 2216 557 2249; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_137' title='bbox 572 2205 749 2249; x_wconf 94' lang='ell' dir='ltr'>πρόθεση</span> <span class='ocrx_word' id='word_1_138' title='bbox 767 2205 941 2240; x_wconf 96' lang='ell' dir='ltr'>σκότωσε</span> <span class='ocrx_word' id='word_1_139' title='bbox 955 2216 998 2240; x_wconf 96' lang='ell' dir='ltr'>την</span> <span class='ocrx_word' id='word_1_140' title='bbox 1014 2205 1122 2240; x_wconf 90' lang='ell' dir='ltr'>κόρη</span> <span class='ocrx_word' id='word_1_141' title='bbox 1133 2216 1198 2249; x_wconf 98' lang='ell' dir='ltr'>του</span> <span class='ocrx_word' id='word_1_142' title='bbox 1215 2205 1309 2240; x_wconf 90' lang='ell' dir='ltr'>κατά</span> <span class='ocrx_word' id='word_1_143' title='bbox 1326 2205 1345 2249; x_wconf 98' lang='ell' dir='ltr'>ή</span> <span class='ocrx_word' id='word_1_144' title='bbox 1364 2205 1456 2249; x_wconf 91' lang='ell' dir='ltr'>μετά</span> <span class='ocrx_word' id='word_1_145' title='bbox 1469 2216 1536 2240; x_wconf 96' lang='ell' dir='ltr'>τον</span> <span class='ocrx_word' id='word_1_146' title='bbox 1548 2205 1694 2247; x_wconf 95' lang='ell' dir='ltr'>τοκετό,</span> <span class='ocrx_word' id='word_1_147' title='bbox 1711 2205 1810 2240; x_wconf 90' lang='ell' dir='ltr'>αλλά</span> <span class='ocrx_word' id='word_1_148' title='bbox 1825 2205 1901 2240; x_wconf 93' lang='ell' dir='ltr'>ενώ</span> <span class='ocrx_word' id='word_1_149' title='bbox 1917 2205 2238 2249; x_wconf 90' lang='ell' dir='ltr'>εξακολουθούσε</span>
</span>
<span class='ocr_line' id='line_1_22' title="bbox 242 2266 2077 2310; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_150' title='bbox 242 2266 368 2310; x_wconf 90' lang='ell' dir='ltr'>ακόμη</span> <span class='ocrx_word' id='word_1_151' title='bbox 387 2277 406 2310; x_wconf 98' lang='ell' dir='ltr'>η</span> <span class='ocrx_word' id='word_1_152' title='bbox 424 2266 633 2310; x_wconf 90' lang='ell' dir='ltr'>διατάραξη</span> <span class='ocrx_word' id='word_1_153' title='bbox 648 2277 717 2301; x_wconf 95' lang='ell' dir='ltr'>του</span> <span class='ocrx_word' id='word_1_154' title='bbox 734 2266 974 2310; x_wconf 90' lang='ell' dir='ltr'>οργανισμού</span> <span class='ocrx_word' id='word_1_155' title='bbox 989 2277 1054 2310; x_wconf 98' lang='ell' dir='ltr'>της</span> <span class='ocrx_word' id='word_1_156' title='bbox 1069 2266 1147 2301; x_wconf 90' lang='ell' dir='ltr'>από</span> <span class='ocrx_word' id='word_1_157' title='bbox 1163 2266 1293 2308; x_wconf 90' lang='ell' dir='ltr'>αυτόν,</span> <span class='ocrx_word' id='word_1_158' title='bbox 1308 2266 1527 2310; x_wconf 90' lang='ell' dir='ltr'>τιμωρείται</span> <span class='ocrx_word' id='word_1_159' title='bbox 1542 2277 1585 2310; x_wconf 91' lang='ell' dir='ltr'>με</span> <span class='ocrx_word' id='word_1_160' title='bbox 1604 2266 1781 2310; x_wconf 93' lang='ell' dir='ltr'>κάθειρξη</span> <span class='ocrx_word' id='word_1_161' title='bbox 1799 2266 1874 2310; x_wconf 98' lang='ell' dir='ltr'>έως</span> <span class='ocrx_word' id='word_1_162' title='bbox 1889 2266 1986 2301; x_wconf 90' lang='ell' dir='ltr'>δέκα</span> <span class='ocrx_word' id='word_1_163' title='bbox 2000 2266 2077 2310; x_wconf 98' lang='ell' dir='ltr'>έτη.</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_12' title="bbox 959 2449 1521 2493">
<p class='ocr_par' dir='ltr' id='par_1_12' title="bbox 959 2449 1521 2493">
<span class='ocr_line' id='line_1_23' title="bbox 959 2449 1521 2493; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_164' title='bbox 959 2452 993 2484; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_165' title='bbox 1011 2449 1240 2493; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_166' title='bbox 1253 2460 1323 2485; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_167' title='bbox 1340 2449 1521 2493; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_13' title="bbox 1030 2571 1450 2676">
<p class='ocr_par' dir='ltr' id='par_1_13' title="bbox 1030 2571 1450 2676">
<span class='ocr_line' id='line_1_24' title="bbox 1129 2571 1352 2615; baseline 0 -8; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_168' title='bbox 1129 2571 1264 2615; x_wconf 99' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_169' title='bbox 1279 2574 1352 2607; x_wconf 91' lang='ell'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_25' title="bbox 1030 2632 1450 2676; baseline 0.002 -9; x_size 44; x_descenders 8; x_ascenders 11"><span class='ocrx_word' id='word_1_170' title='bbox 1030 2632 1206 2676; x_wconf 94' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_171' title='bbox 1220 2643 1286 2676; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_172' title='bbox 1302 2632 1450 2676; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_14' title="bbox 240 2754 2240 3104">
<p class='ocr_par' dir='ltr' id='par_1_14' title="bbox 242 2754 2240 2859">
<span class='ocr_line' id='line_1_26' title="bbox 244 2754 2240 2798; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_173' title='bbox 244 2757 274 2789; x_wconf 93' lang='ell'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_174' title='bbox 292 2754 440 2798; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_175' title='bbox 457 2754 572 2798; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_176' title='bbox 589 2765 632 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_177' title='bbox 653 2754 864 2798; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_178' title='bbox 884 2765 949 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_179' title='bbox 968 2754 1111 2798; x_wconf 87' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_180' title='bbox 1133 2754 1332 2789; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_181' title='bbox 1347 2765 1415 2798; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_182' title='bbox 1435 2754 1558 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_183' title='bbox 1577 2765 1643 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_184' title='bbox 1660 2754 1879 2798; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_185' title='bbox 1899 2765 1942 2798; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_186' title='bbox 1965 2754 2142 2798; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_187' title='bbox 2166 2754 2240 2798; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_27' title="bbox 242 2815 430 2859; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_188' title='bbox 242 2815 339 2850; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_189' title='bbox 353 2815 430 2859; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span>
</span>
</p>
<p class='ocr_par' dir='ltr' id='par_1_15' title="bbox 240 2876 2237 3104">
<span class='ocr_line' id='line_1_28' title="bbox 243 2876 2235 2920; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_190' title='bbox 243 2879 274 2911; x_wconf 93' lang='ell'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_191' title='bbox 302 2876 449 2920; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_192' title='bbox 480 2887 524 2920; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_193' title='bbox 552 2887 595 2920; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_194' title='bbox 626 2876 837 2920; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_195' title='bbox 866 2887 931 2920; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_196' title='bbox 960 2876 1102 2920; x_wconf 88' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_197' title='bbox 1136 2876 1155 2920; x_wconf 98' lang='ell' dir='ltr'>&#9679;</span> <span class='ocrx_word' id='word_1_198' title='bbox 1184 2888 1259 2911; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_199' title='bbox 1287 2876 1510 2920; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_200' title='bbox 1538 2887 1614 2911; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_201' title='bbox 1645 2876 1761 2920; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_202' title='bbox 1788 2887 1831 2920; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_203' title='bbox 1861 2876 1990 2920; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_204' title='bbox 2024 2876 2186 2920; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_205' title='bbox 2216 2876 2235 2920; x_wconf 98' lang='ell' dir='ltr'>&#9679;</span>
</span>
<span class='ocr_line' id='line_1_29' title="bbox 242 2937 2236 2981; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_206' title='bbox 242 2937 441 2981; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_207' title='bbox 454 2948 520 2981; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_208' title='bbox 535 2948 582 2972; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_209' title='bbox 598 2937 693 2981; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_210' title='bbox 711 2937 810 2972; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_211' title='bbox 823 2937 982 2981; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_212' title='bbox 999 2948 1048 2972; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_213' title='bbox 1063 2937 1296 2979; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_214' title='bbox 1313 2937 1512 2972; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_215' title='bbox 1523 2948 1591 2981; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_216' title='bbox 1606 2948 1695 2981; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_217' title='bbox 1703 2937 1851 2981; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_218' title='bbox 1867 2948 1943 2981; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_219' title='bbox 1958 2937 2177 2981; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_220' title='bbox 2193 2948 2236 2981; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_30' title="bbox 242 2998 2237 3042; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_221' title='bbox 242 2998 437 3042; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_222' title='bbox 471 2998 546 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_223' title='bbox 576 2998 661 3042; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_224' title='bbox 693 2998 756 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_225' title='bbox 793 2998 812 3042; x_wconf 98' lang='ell' dir='ltr'>&#9679;</span> <span class='ocrx_word' id='word_1_226' title='bbox 844 2998 1052 3042; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_227' title='bbox 1085 2998 1197 3042; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_228' title='bbox 1233 3009 1294 3033; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_229' title='bbox 1324 3009 1372 3033; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_230' title='bbox 1404 2998 1554 3042; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_231' title='bbox 1586 2999 1662 3033; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_232' title='bbox 1696 2998 1930 3042; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_233' title='bbox 1966 3009 2009 3042; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_234' title='bbox 2042 2998 2237 3042; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span>
</span>
<span class='ocr_line' id='line_1_31' title="bbox 240 3060 2235 3104; baseline 0 -9; x_size 44; x_descenders 9; x_ascenders 11"><span class='ocrx_word' id='word_1_235' title='bbox 240 3060 493 3104; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_236' title='bbox 516 3060 591 3095; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_237' title='bbox 615 3060 712 3095; x_wconf 93' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_238' title='bbox 736 3071 797 3095; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_239' title='bbox 817 3060 1025 3104; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_240' title='bbox 1050 3060 1175 3104; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_241' title='bbox 1202 3063 1261 3095; x_wconf 89' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_242' title='bbox 1284 3071 1351 3104; x_wconf 96' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_243' title='bbox 1376 3060 1452 3095; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_244' title='bbox 1474 3060 1599 3104; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_245' title='bbox 1627 3060 1817 3104; x_wconf 92' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_246' title='bbox 1845 3060 1938 3095; x_wconf 90' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_247' title='bbox 1959 3071 2003 3095; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_248' title='bbox 2029 3060 2144 3104; x_wconf 91' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;&#9679;&#9679;</span> <span class='ocrx_word' id='word_1_249' title='bbox 2166 3071 2235 3095; x_wconf 95' lang='ell' dir='ltr'>&#9679;&#9679;&#9679;</span>
</span>
</p>
</div>
<div class='ocr_carea' id='block_1_15' title="bbox 2193 3257 2238 3292">
<p class='ocr_par' dir='ltr' id='par_1_16' title="bbox 2193 3257 2238 3292">
<span class='ocr_line' id='line_1_32' title="bbox 2193 3257 2238 3292; baseline 0 0; x_size 46.666668; x_descenders 11.666667; x_ascenders 11.666667"><span class='ocrx_word' id='word_1_250' title='bbox 2193 3257 2238 3292; x_wconf 88' lang='ell'>78</span>
</span>
</p>
</div>
</div>
</body>
</html>

View File

@ -0,0 +1,518 @@
#!/usr/bin/env python3
#
# Copyright (c) 2010, Jonathan Brinley
# Original version from: https://github.com/jbrinley/HocrConverter
#
# Copyright (c) 2013-14, Julien Pfefferkorn
# Modifications
#
# Copyright (c) 2015-16, James R. Barlow
# Set text to transparent
# Copyright (c) 2022, WordMord & Alex Roidl
# Set text back to visible and change bounding boxes
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the
# "Software"), to deal in the Software without restriction, including
# without limitation the rights to use, copy, modify, merge, publish,
# distribute, sublicense, and/or sell copies of the Software, and to
# permit persons to whom the Software is furnished to do so, subject to
# the following conditions:
#
# The above copyright notice and this permission notice shall be included
# in all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
# OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
# IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
# CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
# TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
# SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
import argparse
import os
import re
from itertools import chain
from math import atan, cos, sin
from pathlib import Path
from typing import Any, NamedTuple, Optional, Tuple, Union
from xml.etree import ElementTree
from reportlab.lib.colors import black, cyan, magenta, red
from reportlab.lib.units import inch
from reportlab.pdfgen.canvas import Canvas
from reportlab.pdfbase import pdfmetrics
from reportlab.pdfbase.ttfonts import TTFont
from reportlab.pdfbase.pdfmetrics import registerFontFamily
# According to Wikipedia these languages are supported in the ISO-8859-1 character
# set, meaning reportlab can generate them and they are compatible with hocr,
# assuming Tesseract has the necessary languages installed. Note that there may
# not be language packs for them.
HOCR_OK_LANGS = frozenset(
[
# Languages fully covered by Latin-1:
'afr', # Afrikaans
'alb', # Albanian
'ast', # Leonese
'baq', # Basque
'bre', # Breton
'cos', # Corsican
'eng', # English
'eus', # Basque
'fao', # Faoese
'gla', # Scottish Gaelic
'glg', # Galician
'glv', # Manx
'ice', # Icelandic
'ind', # Indonesian
'isl', # Icelandic
'ita', # Italian
'ltz', # Luxembourgish
'mal', # Malay Rumi
'mga', # Irish
'nor', # Norwegian
'oci', # Occitan
'por', # Portugeuse
'roh', # Romansh
'sco', # Scots
'sma', # Sami
'spa', # Spanish
'sqi', # Albanian
'swa', # Swahili
'swe', # Swedish
'tgl', # Tagalog
'wln', # Walloon
# Languages supported by Latin-1 except for a few rare characters that OCR
# is probably not trained to recognize anyway:
'cat', # Catalan
'cym', # Welsh
'dan', # Danish
'deu', # German
'dut', # Dutch
'est', # Estonian
'fin', # Finnish
'fra', # French
'hun', # Hungarian
'kur', # Kurdish
'nld', # Dutch
'wel', # Welsh
]
)
Element = ElementTree.Element
class Rect(NamedTuple): # pylint: disable=inherit-non-class
"""A rectangle for managing PDF coordinates."""
x1: Any
y1: Any
x2: Any
y2: Any
class HocrTransformError(Exception):
pass
class HocrTransform:
"""
A class for converting documents from the hOCR format.
For details of the hOCR format, see:
http://kba.cloud/hocr-spec/
"""
box_pattern = re.compile(r'bbox((\s+\d+){4})')
baseline_pattern = re.compile(
r'''
baseline \s+
([\-\+]?\d*\.?\d*) \s+ # +/- decimal float
([\-\+]?\d+) # +/- int''',
re.VERBOSE,
)
ligatures = str.maketrans(
{'': 'ff', '': 'ffi', '': 'ffl', '': 'fi', '': 'fl'}
)
def __init__(self, *, hocr_filename: Union[str, Path], dpi: float):
self.dpi = dpi
self.hocr = ElementTree.parse(os.fspath(hocr_filename))
# if the hOCR file has a namespace, ElementTree requires its use to
# find elements
matches = re.match(r'({.*})html', self.hocr.getroot().tag)
self.xmlns = ''
if matches:
self.xmlns = matches.group(1)
# get dimension in pt (not pixel!!!!) of the OCRed image
self.width, self.height = None, None
for div in self.hocr.findall(self._child_xpath('div', 'ocr_page')):
coords = self.element_coordinates(div)
pt_coords = self.pt_from_pixel(coords)
self.width = pt_coords.x2 - pt_coords.x1
self.height = pt_coords.y2 - pt_coords.y1
# there shouldn't be more than one, and if there is, we don't want
# it
break
if self.width is None or self.height is None:
raise HocrTransformError("hocr file is missing page dimensions")
def __str__(self): # pragma: no cover
"""
Return the textual content of the HTML body
"""
if self.hocr is None:
return ''
body = self.hocr.find(self._child_xpath('body'))
if body:
return self._get_element_text(body)
else:
return ''
def _get_element_text(self, element: Element):
"""
Return the textual content of the element and its children
"""
text = ''
if element.text is not None:
text += element.text
for child in element:
text += self._get_element_text(child)
if element.tail is not None:
text += element.tail
return text
@classmethod
def element_coordinates(cls, element: Element) -> Rect:
"""
Returns a tuple containing the coordinates of the bounding box around
an element
"""
out = Rect._make(0 for _ in range(4))
if 'title' in element.attrib:
matches = cls.box_pattern.search(element.attrib['title'])
if matches:
coords = matches.group(1).split()
out = Rect._make(int(coords[n]) for n in range(4))
return out
@classmethod
def baseline(cls, element: Element) -> Tuple[float, float]:
"""
Returns a tuple containing the baseline slope and intercept.
"""
if 'title' in element.attrib:
matches = cls.baseline_pattern.search(element.attrib['title'])
if matches:
return float(matches.group(1)), int(matches.group(2))
return (0.0, 0.0)
def pt_from_pixel(self, pxl) -> Rect:
"""
Returns the quantity in PDF units (pt) given quantity in pixels
"""
return Rect._make((c / self.dpi * inch) for c in pxl)
def _child_xpath(self, html_tag: str, html_class: Optional[str] = None) -> str:
xpath = f".//{self.xmlns}{html_tag}"
if html_class:
xpath += f"[@class='{html_class}']"
return xpath
@classmethod
def replace_unsupported_chars(cls, s: str) -> str:
"""
Given an input string, returns the corresponding string that:
* is available in the Helvetica facetype
* does not contain any ligature (to allow easy search in the PDF file)
"""
return s.translate(cls.ligatures)
def topdown_position(self, element):
pxl_line_coords = self.element_coordinates(element)
line_box = self.pt_from_pixel(pxl_line_coords)
# Coordinates here are still in the hocr coordinate system, so 0 on the y axis
# is the top of the page and increasing values of y will move towards the
# bottom of the page.
return line_box.y2
def to_pdf(
self,
*,
out_filename: Path,
image_filename: Optional[Path] = None,
show_bounding_boxes: bool = False,
fontname: str = "Helvetica",
invisible_text: bool = False,
interword_spaces: bool = False,
) -> None:
"""
Creates a PDF file with an image superimposed on top of the text.
Text is positioned according to the bounding box of the lines in
the hOCR file.
The image need not be identical to the image used to create the hOCR
file.
It can have a lower resolution, different color mode, etc.
Arguments:
out_filename: Path of PDF to write.
image_filename: Image to use for this file. If omitted, the OCR text
is shown.
show_bounding_boxes: Show bounding boxes around various text regions,
for debugging.
fontname: Name of font to use.
invisible_text: If True, text is rendered invisible so that is
selectable but never drawn. If False, text is visible and may
be seen if the image is skipped or deleted in Acrobat.
interword_spaces: If True, insert spaces between words rather than
drawing each word without spaces. Generally this improves text
extraction.
"""
# create the PDF file
# page size in points (1/72 in.)
pdfmetrics.registerFont(TTFont('Greek', '../styles/fonts/greek.ttf'))
pdfmetrics.registerFont(TTFont('GreekB', '../styles/fonts/greek-bold.ttf'))
registerFontFamily('Greek', normal='Greek', bold='GreekB')
pdf = Canvas(
os.fspath(out_filename),
pagesize=(self.width, self.height),
pageCompression=1,
)
if image_filename is not None:
pdf.drawImage(
os.fspath(image_filename), 0, 0, width=self.width, height=self.height
)
# draw bounding box for each paragraph
# light blue for bounding box of paragraph
pdf.setStrokeColor(black)
# light blue for bounding box of paragraph
pdf.setFillColor(black)
pdf.setLineWidth(1) # no line for bounding box
for elem in self.hocr.iterfind(self._child_xpath('p', 'ocr_par')):
elemtxt = self._get_element_text(elem).rstrip()
if len(elemtxt) == 0:
continue
pxl_coords = self.element_coordinates(elem)
pt = self.pt_from_pixel(pxl_coords)
# draw the bbox border
if show_bounding_boxes: # pragma: no cover
pdf.rect(
pt.x1, self.height - pt.y2, pt.x2 - pt.x1, pt.y2 - pt.y1, fill=1
)
found_lines = False
for line in sorted(
chain(
self.hocr.iterfind(self._child_xpath('span', 'ocr_header')),
self.hocr.iterfind(self._child_xpath('span', 'ocr_line')),
self.hocr.iterfind(self._child_xpath('span', 'ocr_textfloat')),
),
key=self.topdown_position,
):
found_lines = True
self._do_line(
pdf,
line,
"ocrx_word",
fontname,
invisible_text,
interword_spaces,
show_bounding_boxes,
)
if not found_lines:
# Tesseract did not report any lines (just words)
root = self.hocr.find(self._child_xpath('div', 'ocr_page'))
self._do_line(
pdf,
root,
"ocrx_word",
fontname,
invisible_text,
interword_spaces,
show_bounding_boxes,
)
# put the image on the page, scaled to fill the page
# finish up the page and save it
pdf.showPage()
pdf.save()
@classmethod
def polyval(cls, poly, x): # pragma: no cover
return x * poly[0] + poly[1]
def _do_line(
self,
pdf: Canvas,
line: Optional[Element],
elemclass: str,
fontname: str,
invisible_text: bool,
interword_spaces: bool,
show_bounding_boxes: bool,
):
if not line:
return
pxl_line_coords = self.element_coordinates(line)
line_box = self.pt_from_pixel(pxl_line_coords)
line_height = line_box.y2 - line_box.y1
slope, pxl_intercept = self.baseline(line)
if abs(slope) < 0.005:
slope = 0.0
angle = atan(slope)
cos_a, sin_a = cos(angle), sin(angle)
text = pdf.beginText()
intercept = pxl_intercept / self.dpi * inch
# Don't allow the font to break out of the bounding box. Division by
# cos_a accounts for extra clearance between the glyph's vertical axis
# on a sloped baseline and the edge of the bounding box.
fontsize = (line_height - abs(intercept)) / cos_a * 1.2
#fontsize = 10.5
text.setFont('Greek', fontsize)
#if invisible_text:
# text.setTextRenderMode(3) # Invisible (indicates OCR text)
# Intercept is normally negative, so this places it above the bottom
# of the line box
baseline_y2 = self.height - (line_box.y2 + intercept)
if False: # pragma: no cover
# draw the baseline in magenta, dashed
pdf.setDash()
pdf.setStrokeColor(magenta)
pdf.setLineWidth(0.5)
# negate slope because it is defined as a rise/run in pixel
# coordinates and page coordinates have the y axis flipped
pdf.line(
line_box.x1,
baseline_y2,
line_box.x2,
self.polyval((-slope, baseline_y2), line_box.x2 - line_box.x1),
)
# light green for bounding box of word/line
pdf.setDash(6, 3)
pdf.setStrokeColor(red)
#text.setTextTransform(cos_a, -sin_a, sin_a, cos_a, line_box.x1, baseline_y2)
text.setTextOrigin(line_box.x1, baseline_y2)
##pdf.translate(line_box.x1, baseline_y2)
pdf.setFillColor(black) # text in black
elements = line.findall(self._child_xpath('span', elemclass))
for elem in elements:
elemtxt = self._get_element_text(elem).strip()
elemtxt = self.replace_unsupported_chars(elemtxt)
if elemtxt == '':
continue
pxl_coords = self.element_coordinates(elem)
box = self.pt_from_pixel(pxl_coords)
if False:
# if `--interword-spaces` is true, append a space
# to the end of each text element to allow simpler PDF viewers
# such as PDF.js to better recognize words in search and copy
# and paste. Do not remove space from last word in line, even
# though it would look better, because it will interfere with
# naive text extraction. \n does not work either.
elemtxt += ' '
box = Rect._make(
(
box.x1,
line_box.y1,
box.x2 + pdf.stringWidth(' ', fontname, line_height),
line_box.y2,
)
)
box_width = box.x2 - box.x1
font_width = pdf.stringWidth(elemtxt, fontname, fontsize)
# draw the bbox border
if False: # pragma: no cover
pdf.rect(
box.x1, self.height - line_box.y2, box_width, line_height, fill=0
)
# Adjust relative position of cursor
# This is equivalent to:
# text.setTextOrigin(pt.x1, self.height - line_box.y2)
# but the former generates a full text reposition matrix (Tm) in the
# content stream while this issues a "offset" (Td) command.
# .moveCursor() is relative to start of the text line, where the
# "text line" means whatever reportlab defines it as. Do not use
# use .getCursor(), since moveCursor() rather unintuitively plans
# its moves relative to .getStartOfLine().
# For skewed lines, in the text transform we set up a rotated
# coordinate system, so we don't have to account for the
# incremental offset. Surprisingly most PDF viewers can handle this.
cursor = text.getStartOfLine()
dx = box.x1 - cursor[0]
dy = baseline_y2 - cursor[1]
text.moveCursor(dx, dy)
# If reportlab tells us this word is 0 units wide, our best seems
# to be to suppress this text
if font_width > 0:
#text.setHorizScale(100 * box_width / font_width)
text.textOut(elemtxt)
pdf.drawText(text)
if __name__ == "__main__":
parser = argparse.ArgumentParser(description='Convert hocr file to PDF')
parser.add_argument(
'-b',
'--boundingboxes',
action="store_true",
default=False,
help='Show bounding boxes borders',
)
parser.add_argument(
'-r',
'--resolution',
type=int,
default=300,
help='Resolution of the image that was OCRed',
)
parser.add_argument(
'-i',
'--image',
default=None,
help='Path to the image to be placed above the text',
)
parser.add_argument(
'--interword-spaces',
action='store_true',
default=False,
help='Add spaces between words',
)
parser.add_argument('hocrfile', help='Path to the hocr file to be parsed')
parser.add_argument('outputfile', help='Path to the PDF file to be generated')
args = parser.parse_args()
hocr = HocrTransform(hocr_filename=args.hocrfile, dpi=args.resolution)
hocr.to_pdf(
out_filename=args.outputfile,
image_filename=args.image,
show_bounding_boxes=args.boundingboxes,
interword_spaces=args.interword_spaces,
)

BIN
scripts/images/anthropoktonia.png Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 406 KiB

BIN
scripts/images/blank.png Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

65
scripts/metalucktria.py Normal file
View File

@ -0,0 +1,65 @@
import random
import time
wordmord = {
'death':
{
'para-etymology': ['murder', 'killing', 'φόνος', 'φονός', 'φώνος', 'φωνή', 'διακοπή', 'depth', 'debt', 'θα είναι αυτός', 'θα νάτος'],
'demonic-etymology': ['telos', 'kako', 'symfora', 'tragodia', 'καλά του κάνανε'],
'pira-etymology': ['most', 'horismos', 'siopi', 'έφυγε', 'την χάσαμε', 'ταξίδεψε', 'θάνατος']
},
'kollisa':
{
'para-etymology': ['kelossi', 'attachment', 'virus', 'friendship', 'obsession', 'crash', 'petrified'],
'demonic-etymology': ['disease', 'infection', 'τον κόλησαν'],
'pira-etymology': ['ευθύνη', 'πήρα την ευθύνη', 'συναίνεση']
},
'monster':
{
'para-etymology': ['κούκλα', 'queen', 'tsoula', 'περήφανη', 'τρελλή', 'τσουλάρα', 'chic'],
'demonic-etymology': ['τέρας', 'φρικιό', 'πούστης', 'αδελφή', 'τέτοιος', 'ληστής', 'πρεζόνι'],
'pira-etymology': ['είμαι τσούλα', 'είμαι τσουλάρα']
}
}
#print(wordmord['death']['para-etymology'][0])
sentence = input('give me your words / δώσε μου κείμενο: ')
def makedemonic():
new_sentence = sentence
for word in wordmord:
if word in new_sentence:
new_sentence = new_sentence.replace(word, random.choice(wordmord[word]['demonic-etymology']))
print(new_sentence)
def makepara():
new_sentence = sentence
for word in wordmord:
if word in new_sentence:
new_sentence = new_sentence.replace(word, random.choice(wordmord[word]['para-etymology']))
print(new_sentence)
def makepira():
new_sentence = sentence
for word in wordmord:
if word in new_sentence:
new_sentence = new_sentence.replace(word, random.choice(wordmord[word]['pira-etymology']))
print(new_sentence)
type = input('choose type of transformation / τύπος μετάλλαξης: ')
if type == 'demonic':
makedemonic()
elif type == 'para':
makepara()
elif type == 'pira':
makepira()
else:
makedemonic()
time.sleep(1)
makepara()
time.sleep(1)
makepira()

Binary file not shown.

76
scripts/venv/bin/activate Normal file
View File

@ -0,0 +1,76 @@
# This file must be used with "source bin/activate" *from bash*
# you cannot run it directly
deactivate () {
# reset old environment variables
if [ -n "${_OLD_VIRTUAL_PATH:-}" ] ; then
PATH="${_OLD_VIRTUAL_PATH:-}"
export PATH
unset _OLD_VIRTUAL_PATH
fi
if [ -n "${_OLD_VIRTUAL_PYTHONHOME:-}" ] ; then
PYTHONHOME="${_OLD_VIRTUAL_PYTHONHOME:-}"
export PYTHONHOME
unset _OLD_VIRTUAL_PYTHONHOME
fi
# This should detect bash and zsh, which have a hash command that must
# be called to get it to forget past commands. Without forgetting
# past commands the $PATH changes we made may not be respected
if [ -n "${BASH:-}" -o -n "${ZSH_VERSION:-}" ] ; then
hash -r
fi
if [ -n "${_OLD_VIRTUAL_PS1:-}" ] ; then
PS1="${_OLD_VIRTUAL_PS1:-}"
export PS1
unset _OLD_VIRTUAL_PS1
fi
unset VIRTUAL_ENV
if [ ! "$1" = "nondestructive" ] ; then
# Self destruct!
unset -f deactivate
fi
}
# unset irrelevant variables
deactivate nondestructive
VIRTUAL_ENV="/var/www/wordmord/scripts/venv"
export VIRTUAL_ENV
_OLD_VIRTUAL_PATH="$PATH"
PATH="$VIRTUAL_ENV/bin:$PATH"
export PATH
# unset PYTHONHOME if set
# this will fail if PYTHONHOME is set to the empty string (which is bad anyway)
# could use `if (set -u; : $PYTHONHOME) ;` in bash
if [ -n "${PYTHONHOME:-}" ] ; then
_OLD_VIRTUAL_PYTHONHOME="${PYTHONHOME:-}"
unset PYTHONHOME
fi
if [ -z "${VIRTUAL_ENV_DISABLE_PROMPT:-}" ] ; then
_OLD_VIRTUAL_PS1="${PS1:-}"
if [ "x(venv) " != x ] ; then
PS1="(venv) ${PS1:-}"
else
if [ "`basename \"$VIRTUAL_ENV\"`" = "__" ] ; then
# special case for Aspen magic directories
# see http://www.zetadev.com/software/aspen/
PS1="[`basename \`dirname \"$VIRTUAL_ENV\"\``] $PS1"
else
PS1="(`basename \"$VIRTUAL_ENV\"`)$PS1"
fi
fi
export PS1
fi
# This should detect bash and zsh, which have a hash command that must
# be called to get it to forget past commands. Without forgetting
# past commands the $PATH changes we made may not be respected
if [ -n "${BASH:-}" -o -n "${ZSH_VERSION:-}" ] ; then
hash -r
fi

View File

@ -0,0 +1,37 @@
# This file must be used with "source bin/activate.csh" *from csh*.
# You cannot run it directly.
# Created by Davide Di Blasi <davidedb@gmail.com>.
# Ported to Python 3.3 venv by Andrew Svetlov <andrew.svetlov@gmail.com>
alias deactivate 'test $?_OLD_VIRTUAL_PATH != 0 && setenv PATH "$_OLD_VIRTUAL_PATH" && unset _OLD_VIRTUAL_PATH; rehash; test $?_OLD_VIRTUAL_PROMPT != 0 && set prompt="$_OLD_VIRTUAL_PROMPT" && unset _OLD_VIRTUAL_PROMPT; unsetenv VIRTUAL_ENV; test "\!:*" != "nondestructive" && unalias deactivate'
# Unset irrelevant variables.
deactivate nondestructive
setenv VIRTUAL_ENV "/var/www/wordmord/scripts/venv"
set _OLD_VIRTUAL_PATH="$PATH"
setenv PATH "$VIRTUAL_ENV/bin:$PATH"
set _OLD_VIRTUAL_PROMPT="$prompt"
if (! "$?VIRTUAL_ENV_DISABLE_PROMPT") then
if ("venv" != "") then
set env_name = "venv"
else
if (`basename "VIRTUAL_ENV"` == "__") then
# special case for Aspen magic directories
# see http://www.zetadev.com/software/aspen/
set env_name = `basename \`dirname "$VIRTUAL_ENV"\``
else
set env_name = `basename "$VIRTUAL_ENV"`
endif
endif
set prompt = "[$env_name] $prompt"
unset env_name
endif
alias pydoc python -m pydoc
rehash

View File

@ -0,0 +1,75 @@
# This file must be used with ". bin/activate.fish" *from fish* (http://fishshell.org)
# you cannot run it directly
function deactivate -d "Exit virtualenv and return to normal shell environment"
# reset old environment variables
if test -n "$_OLD_VIRTUAL_PATH"
set -gx PATH $_OLD_VIRTUAL_PATH
set -e _OLD_VIRTUAL_PATH
end
if test -n "$_OLD_VIRTUAL_PYTHONHOME"
set -gx PYTHONHOME $_OLD_VIRTUAL_PYTHONHOME
set -e _OLD_VIRTUAL_PYTHONHOME
end
if test -n "$_OLD_FISH_PROMPT_OVERRIDE"
functions -e fish_prompt
set -e _OLD_FISH_PROMPT_OVERRIDE
functions -c _old_fish_prompt fish_prompt
functions -e _old_fish_prompt
end
set -e VIRTUAL_ENV
if test "$argv[1]" != "nondestructive"
# Self destruct!
functions -e deactivate
end
end
# unset irrelevant variables
deactivate nondestructive
set -gx VIRTUAL_ENV "/var/www/wordmord/scripts/venv"
set -gx _OLD_VIRTUAL_PATH $PATH
set -gx PATH "$VIRTUAL_ENV/bin" $PATH
# unset PYTHONHOME if set
if set -q PYTHONHOME
set -gx _OLD_VIRTUAL_PYTHONHOME $PYTHONHOME
set -e PYTHONHOME
end
if test -z "$VIRTUAL_ENV_DISABLE_PROMPT"
# fish uses a function instead of an env var to generate the prompt.
# save the current fish_prompt function as the function _old_fish_prompt
functions -c fish_prompt _old_fish_prompt
# with the original prompt function renamed, we can override with our own.
function fish_prompt
# Save the return status of the last command
set -l old_status $status
# Prompt override?
if test -n "(venv) "
printf "%s%s" "(venv) " (set_color normal)
else
# ...Otherwise, prepend env
set -l _checkbase (basename "$VIRTUAL_ENV")
if test $_checkbase = "__"
# special case for Aspen magic directories
# see http://www.zetadev.com/software/aspen/
printf "%s[%s]%s " (set_color -b blue white) (basename (dirname "$VIRTUAL_ENV")) (set_color normal)
else
printf "%s(%s)%s" (set_color -b blue white) (basename "$VIRTUAL_ENV") (set_color normal)
end
end
# Restore the return status of the previous command.
echo "exit $old_status" | .
_old_fish_prompt
end
set -gx _OLD_FISH_PROMPT_OVERRIDE "$VIRTUAL_ENV"
end

8
scripts/venv/bin/flask Executable file
View File

@ -0,0 +1,8 @@
#!/var/www/wordmord/scripts/venv/bin/python3
# -*- coding: utf-8 -*-
import re
import sys
from flask.cli import main
if __name__ == '__main__':
sys.argv[0] = re.sub(r'(-script\.pyw|\.exe)?$', '', sys.argv[0])
sys.exit(main())

11
scripts/venv/bin/pip Executable file
View File

@ -0,0 +1,11 @@
#!/var/www/wordmord/scripts/venv/bin/python3
# -*- coding: utf-8 -*-
import re
import sys
from pip._internal.cli.main import main
if __name__ == '__main__':
sys.argv[0] = re.sub(r'(-script\.pyw?|\.exe)?$', '', sys.argv[0])
sys.exit(main())

11
scripts/venv/bin/pip3 Executable file
View File

@ -0,0 +1,11 @@
#!/var/www/wordmord/scripts/venv/bin/python3
# -*- coding: utf-8 -*-
import re
import sys
from pip._internal.cli.main import main
if __name__ == '__main__':
sys.argv[0] = re.sub(r'(-script\.pyw?|\.exe)?$', '', sys.argv[0])
sys.exit(main())

11
scripts/venv/bin/pip3.6 Executable file
View File

@ -0,0 +1,11 @@
#!/var/www/wordmord/scripts/venv/bin/python3
# -*- coding: utf-8 -*-
import re
import sys
from pip._internal.cli.main import main
if __name__ == '__main__':
sys.argv[0] = re.sub(r'(-script\.pyw?|\.exe)?$', '', sys.argv[0])
sys.exit(main())

1
scripts/venv/bin/python Symbolic link
View File

@ -0,0 +1 @@
python3

1
scripts/venv/bin/python3 Symbolic link
View File

@ -0,0 +1 @@
/usr/local/bin/python3

View File

@ -0,0 +1,28 @@
Copyright 2010 Pallets
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

View File

@ -0,0 +1,125 @@
Metadata-Version: 2.1
Name: Flask
Version: 2.0.2
Summary: A simple framework for building complex web applications.
Home-page: https://palletsprojects.com/p/flask
Author: Armin Ronacher
Author-email: armin.ronacher@active-4.com
Maintainer: Pallets
Maintainer-email: contact@palletsprojects.com
License: BSD-3-Clause
Project-URL: Donate, https://palletsprojects.com/donate
Project-URL: Documentation, https://flask.palletsprojects.com/
Project-URL: Changes, https://flask.palletsprojects.com/changes/
Project-URL: Source Code, https://github.com/pallets/flask/
Project-URL: Issue Tracker, https://github.com/pallets/flask/issues/
Project-URL: Twitter, https://twitter.com/PalletsTeam
Project-URL: Chat, https://discord.gg/pallets
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Web Environment
Classifier: Framework :: Flask
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Topic :: Internet :: WWW/HTTP :: Dynamic Content
Classifier: Topic :: Internet :: WWW/HTTP :: WSGI
Classifier: Topic :: Internet :: WWW/HTTP :: WSGI :: Application
Classifier: Topic :: Software Development :: Libraries :: Application Frameworks
Requires-Python: >=3.6
Description-Content-Type: text/x-rst
License-File: LICENSE.rst
Requires-Dist: Werkzeug (>=2.0)
Requires-Dist: Jinja2 (>=3.0)
Requires-Dist: itsdangerous (>=2.0)
Requires-Dist: click (>=7.1.2)
Provides-Extra: async
Requires-Dist: asgiref (>=3.2) ; extra == 'async'
Provides-Extra: dotenv
Requires-Dist: python-dotenv ; extra == 'dotenv'
Flask
=====
Flask is a lightweight `WSGI`_ web application framework. It is designed
to make getting started quick and easy, with the ability to scale up to
complex applications. It began as a simple wrapper around `Werkzeug`_
and `Jinja`_ and has become one of the most popular Python web
application frameworks.
Flask offers suggestions, but doesn't enforce any dependencies or
project layout. It is up to the developer to choose the tools and
libraries they want to use. There are many extensions provided by the
community that make adding new functionality easy.
.. _WSGI: https://wsgi.readthedocs.io/
.. _Werkzeug: https://werkzeug.palletsprojects.com/
.. _Jinja: https://jinja.palletsprojects.com/
Installing
----------
Install and update using `pip`_:
.. code-block:: text
$ pip install -U Flask
.. _pip: https://pip.pypa.io/en/stable/getting-started/
A Simple Example
----------------
.. code-block:: python
# save this as app.py
from flask import Flask
app = Flask(__name__)
@app.route("/")
def hello():
return "Hello, World!"
.. code-block:: text
$ flask run
* Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)
Contributing
------------
For guidance on setting up a development environment and how to make a
contribution to Flask, see the `contributing guidelines`_.
.. _contributing guidelines: https://github.com/pallets/flask/blob/main/CONTRIBUTING.rst
Donate
------
The Pallets organization develops and supports Flask and the libraries
it uses. In order to grow the community of contributors and users, and
allow the maintainers to devote more time to the projects, `please
donate today`_.
.. _please donate today: https://palletsprojects.com/donate
Links
-----
- Documentation: https://flask.palletsprojects.com/
- Changes: https://flask.palletsprojects.com/changes/
- PyPI Releases: https://pypi.org/project/Flask/
- Source Code: https://github.com/pallets/flask/
- Issue Tracker: https://github.com/pallets/flask/issues/
- Website: https://palletsprojects.com/p/flask/
- Twitter: https://twitter.com/PalletsTeam
- Chat: https://discord.gg/pallets

View File

@ -0,0 +1,52 @@
../../../bin/flask,sha256=-5QZYaQZbfrPQevySH6wG5RKdwKhG8gb6YsYUFSEZYg,234
Flask-2.0.2.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
Flask-2.0.2.dist-info/LICENSE.rst,sha256=SJqOEQhQntmKN7uYPhHg9-HTHwvY-Zp5yESOf_N9B-o,1475
Flask-2.0.2.dist-info/METADATA,sha256=aKsvjFA_ZjZN1jLh1Ac3aQk-ZUZDPrrwo_TGYW1kdAQ,3839
Flask-2.0.2.dist-info/RECORD,,
Flask-2.0.2.dist-info/REQUESTED,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
Flask-2.0.2.dist-info/WHEEL,sha256=_NOXIqFgOaYmlm9RJLPQZ13BJuEIrp5jx5ptRD5uh3Y,92
Flask-2.0.2.dist-info/entry_points.txt,sha256=gBLA1aKg0OYR8AhbAfg8lnburHtKcgJLDU52BBctN0k,42
Flask-2.0.2.dist-info/top_level.txt,sha256=dvi65F6AeGWVU0TBpYiC04yM60-FX1gJFkK31IKQr5c,6
flask/__init__.py,sha256=9ZCelLoNCpr6eSuLmYlzvbp12B3lrLgoN5U2UWk1vdo,2251
flask/__main__.py,sha256=bYt9eEaoRQWdejEHFD8REx9jxVEdZptECFsV7F49Ink,30
flask/__pycache__/__init__.cpython-36.pyc,,
flask/__pycache__/__main__.cpython-36.pyc,,
flask/__pycache__/app.cpython-36.pyc,,
flask/__pycache__/blueprints.cpython-36.pyc,,
flask/__pycache__/cli.cpython-36.pyc,,
flask/__pycache__/config.cpython-36.pyc,,
flask/__pycache__/ctx.cpython-36.pyc,,
flask/__pycache__/debughelpers.cpython-36.pyc,,
flask/__pycache__/globals.cpython-36.pyc,,
flask/__pycache__/helpers.cpython-36.pyc,,
flask/__pycache__/logging.cpython-36.pyc,,
flask/__pycache__/scaffold.cpython-36.pyc,,
flask/__pycache__/sessions.cpython-36.pyc,,
flask/__pycache__/signals.cpython-36.pyc,,
flask/__pycache__/templating.cpython-36.pyc,,
flask/__pycache__/testing.cpython-36.pyc,,
flask/__pycache__/typing.cpython-36.pyc,,
flask/__pycache__/views.cpython-36.pyc,,
flask/__pycache__/wrappers.cpython-36.pyc,,
flask/app.py,sha256=ectBbi9hGmVHAse5TNcFQZIDRkDAxYUAnLgfuKD0Xws,81975
flask/blueprints.py,sha256=AkAVXZ_MMkjwjklzCAMdBNowTiM0wVQPynnUnXjTL2M,23781
flask/cli.py,sha256=wn2Un9RO32ZfRmCMem5KJ5h62-5lnmy1H9uxgyV-eBs,32238
flask/config.py,sha256=70Uyjh1Jzb9MfTCT7NDhuZWAzyIEu-TIyk6-22MP3zQ,11285
flask/ctx.py,sha256=EM3W0v1ctuFQAGk_HWtQdoJEg_r2f5Le4xcmElxFwwk,17428
flask/debughelpers.py,sha256=W82-xrRmodjopBngI9roYH-q08EbQwN2HEGfDAi6SA0,6184
flask/globals.py,sha256=cWd-R2hUH3VqPhnmQNww892tQS6Yjqg_wg8UvW1M7NM,1723
flask/helpers.py,sha256=00WqA3wYeyjMrnAOPZTUyrnUf7H8ik3CVT0kqGl_qjk,30589
flask/json/__init__.py,sha256=unAKdZBlxMI5OMiTU0-Z2Hl4CF1CMJmqTUzpStiExNw,11822
flask/json/__pycache__/__init__.cpython-36.pyc,,
flask/json/__pycache__/tag.cpython-36.pyc,,
flask/json/tag.py,sha256=fys3HBLssWHuMAIJuTcf2K0bCtosePBKXIWASZEEjnU,8857
flask/logging.py,sha256=1o_hirVGqdj7SBdETnhX7IAjklG89RXlrwz_2CjzQQE,2273
flask/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
flask/scaffold.py,sha256=fM9mRy7QBh9fhJ0VTogVx900dDa5oxz8FOw6OK5F-TU,32796
flask/sessions.py,sha256=Kb7zY4qBIOU2cw1xM5mQ_KmgYUBDFbUYWjlkq0EFYis,15189
flask/signals.py,sha256=H7QwDciK-dtBxinjKpexpglP0E6k0MJILiFWTItfmqU,2136
flask/templating.py,sha256=l96VD39JQ0nue4Bcj7wZ4-FWWs-ppLxvgBCpwDQ4KAk,5626
flask/testing.py,sha256=OsHT-2B70abWH3ulY9IbhLchXIeyj3L-cfcDa88wv5E,10281
flask/typing.py,sha256=hXEVcXoH-QEabmy1F11pYaQ2SonlkMAwfjBAnqj2x18,1982
flask/views.py,sha256=nhq31TRB5Z-z2mjFGZACaaB2Et5XPCmWhWxJxOvLWww,5948
flask/wrappers.py,sha256=VndbHPRBSUUOejmd2Y3ydkoCVUtsS2OJIdJEVIkBVD8,5604

View File

@ -0,0 +1,5 @@
Wheel-Version: 1.0
Generator: bdist_wheel (0.32.3)
Root-Is-Purelib: true
Tag: py3-none-any

View File

@ -0,0 +1,3 @@
[console_scripts]
flask = flask.cli:main

View File

@ -0,0 +1,28 @@
Copyright 2007 Pallets
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

View File

@ -0,0 +1,113 @@
Metadata-Version: 2.1
Name: Jinja2
Version: 3.0.3
Summary: A very fast and expressive template engine.
Home-page: https://palletsprojects.com/p/jinja/
Author: Armin Ronacher
Author-email: armin.ronacher@active-4.com
Maintainer: Pallets
Maintainer-email: contact@palletsprojects.com
License: BSD-3-Clause
Project-URL: Donate, https://palletsprojects.com/donate
Project-URL: Documentation, https://jinja.palletsprojects.com/
Project-URL: Changes, https://jinja.palletsprojects.com/changes/
Project-URL: Source Code, https://github.com/pallets/jinja/
Project-URL: Issue Tracker, https://github.com/pallets/jinja/issues/
Project-URL: Twitter, https://twitter.com/PalletsTeam
Project-URL: Chat, https://discord.gg/pallets
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Web Environment
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Topic :: Internet :: WWW/HTTP :: Dynamic Content
Classifier: Topic :: Text Processing :: Markup :: HTML
Requires-Python: >=3.6
Description-Content-Type: text/x-rst
License-File: LICENSE.rst
Requires-Dist: MarkupSafe (>=2.0)
Provides-Extra: i18n
Requires-Dist: Babel (>=2.7) ; extra == 'i18n'
Jinja
=====
Jinja is a fast, expressive, extensible templating engine. Special
placeholders in the template allow writing code similar to Python
syntax. Then the template is passed data to render the final document.
It includes:
- Template inheritance and inclusion.
- Define and import macros within templates.
- HTML templates can use autoescaping to prevent XSS from untrusted
user input.
- A sandboxed environment can safely render untrusted templates.
- AsyncIO support for generating templates and calling async
functions.
- I18N support with Babel.
- Templates are compiled to optimized Python code just-in-time and
cached, or can be compiled ahead-of-time.
- Exceptions point to the correct line in templates to make debugging
easier.
- Extensible filters, tests, functions, and even syntax.
Jinja's philosophy is that while application logic belongs in Python if
possible, it shouldn't make the template designer's job difficult by
restricting functionality too much.
Installing
----------
Install and update using `pip`_:
.. code-block:: text
$ pip install -U Jinja2
.. _pip: https://pip.pypa.io/en/stable/getting-started/
In A Nutshell
-------------
.. code-block:: jinja
{% extends "base.html" %}
{% block title %}Members{% endblock %}
{% block content %}
<ul>
{% for user in users %}
<li><a href="{{ user.url }}">{{ user.username }}</a></li>
{% endfor %}
</ul>
{% endblock %}
Donate
------
The Pallets organization develops and supports Jinja and other popular
packages. In order to grow the community of contributors and users, and
allow the maintainers to devote more time to the projects, `please
donate today`_.
.. _please donate today: https://palletsprojects.com/donate
Links
-----
- Documentation: https://jinja.palletsprojects.com/
- Changes: https://jinja.palletsprojects.com/changes/
- PyPI Releases: https://pypi.org/project/Jinja2/
- Source Code: https://github.com/pallets/jinja/
- Issue Tracker: https://github.com/pallets/jinja/issues/
- Website: https://palletsprojects.com/p/jinja/
- Twitter: https://twitter.com/PalletsTeam
- Chat: https://discord.gg/pallets

View File

@ -0,0 +1,58 @@
Jinja2-3.0.3.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
Jinja2-3.0.3.dist-info/LICENSE.rst,sha256=O0nc7kEF6ze6wQ-vG-JgQI_oXSUrjp3y4JefweCUQ3s,1475
Jinja2-3.0.3.dist-info/METADATA,sha256=uvKoBSMLvh0qHK-6khEqSe1yOV4jxFzbPSREOp-3BXk,3539
Jinja2-3.0.3.dist-info/RECORD,,
Jinja2-3.0.3.dist-info/WHEEL,sha256=_NOXIqFgOaYmlm9RJLPQZ13BJuEIrp5jx5ptRD5uh3Y,92
Jinja2-3.0.3.dist-info/entry_points.txt,sha256=Qy_DkVo6Xj_zzOtmErrATe8lHZhOqdjpt3e4JJAGyi8,61
Jinja2-3.0.3.dist-info/top_level.txt,sha256=PkeVWtLb3-CqjWi1fO29OCbj55EhX_chhKrCdrVe_zs,7
jinja2/__init__.py,sha256=V3JjnTV-nyIHN6rwj03N1M11fegjGvv-weiHMQwH1pk,2205
jinja2/__pycache__/__init__.cpython-36.pyc,,
jinja2/__pycache__/_identifier.cpython-36.pyc,,
jinja2/__pycache__/async_utils.cpython-36.pyc,,
jinja2/__pycache__/bccache.cpython-36.pyc,,
jinja2/__pycache__/compiler.cpython-36.pyc,,
jinja2/__pycache__/constants.cpython-36.pyc,,
jinja2/__pycache__/debug.cpython-36.pyc,,
jinja2/__pycache__/defaults.cpython-36.pyc,,
jinja2/__pycache__/environment.cpython-36.pyc,,
jinja2/__pycache__/exceptions.cpython-36.pyc,,
jinja2/__pycache__/ext.cpython-36.pyc,,
jinja2/__pycache__/filters.cpython-36.pyc,,
jinja2/__pycache__/idtracking.cpython-36.pyc,,
jinja2/__pycache__/lexer.cpython-36.pyc,,
jinja2/__pycache__/loaders.cpython-36.pyc,,
jinja2/__pycache__/meta.cpython-36.pyc,,
jinja2/__pycache__/nativetypes.cpython-36.pyc,,
jinja2/__pycache__/nodes.cpython-36.pyc,,
jinja2/__pycache__/optimizer.cpython-36.pyc,,
jinja2/__pycache__/parser.cpython-36.pyc,,
jinja2/__pycache__/runtime.cpython-36.pyc,,
jinja2/__pycache__/sandbox.cpython-36.pyc,,
jinja2/__pycache__/tests.cpython-36.pyc,,
jinja2/__pycache__/utils.cpython-36.pyc,,
jinja2/__pycache__/visitor.cpython-36.pyc,,
jinja2/_identifier.py,sha256=EdgGJKi7O1yvr4yFlvqPNEqV6M1qHyQr8Gt8GmVTKVM,1775
jinja2/async_utils.py,sha256=jBcJSmLoQa2PjJdNcOpwaUmBxFNE9rZNwMF7Ob3dP9I,1947
jinja2/bccache.py,sha256=v5rKAlYxIvfJEa0uGzAC6yCYSS3KuXT5Eqi-n9qvNi8,12670
jinja2/compiler.py,sha256=v7zKz-mgSYXmfXD9mRmi2BU0B6Z-1RGZmOXCrsPKzc0,72209
jinja2/constants.py,sha256=GMoFydBF_kdpaRKPoM5cl5MviquVRLVyZtfp5-16jg0,1433
jinja2/debug.py,sha256=r0JL0vfO7HPlyKZEdr6eVlg7HoIg2OQGmJ7SeUEyAeI,8494
jinja2/defaults.py,sha256=boBcSw78h-lp20YbaXSJsqkAI2uN_mD_TtCydpeq5wU,1267
jinja2/environment.py,sha256=Vz20npBX5-SUH_eguQuxrSQDEsLFjho0qcHLdMhY3hA,60983
jinja2/exceptions.py,sha256=ioHeHrWwCWNaXX1inHmHVblvc4haO7AXsjCp3GfWvx0,5071
jinja2/ext.py,sha256=44SjDjeYkkxQTpmC2BetOTxEFMgQ42p2dfSwXmPFcSo,32122
jinja2/filters.py,sha256=jusKTZbd0ddZMaibZkxMUVKNsOsaYtOq_Il8Imtx4BE,52609
jinja2/idtracking.py,sha256=WekexMql3u5n3vDxFsQ_i8HW0j24AtjWTjrPBLWrHww,10721
jinja2/lexer.py,sha256=qNEQqDQw_zO5EaH6rFQsER7Qwn2du0o22prB-TR11HE,29930
jinja2/loaders.py,sha256=1MjXJOU6p4VywFqtpDZhtvtT_vIlmHnZKMKHHw4SZzA,22754
jinja2/meta.py,sha256=GNPEvifmSaU3CMxlbheBOZjeZ277HThOPUTf1RkppKQ,4396
jinja2/nativetypes.py,sha256=KCJl71MogrDih_BHBu6xV5p7Cr_jggAgu-shKTg6L28,3969
jinja2/nodes.py,sha256=i34GPRAZexXMT6bwuf5SEyvdmS-bRCy9KMjwN5O6pjk,34550
jinja2/optimizer.py,sha256=tHkMwXxfZkbfA1KmLcqmBMSaz7RLIvvItrJcPoXTyD8,1650
jinja2/parser.py,sha256=kHnU8v92GwMYkfr0MVakWv8UlSf_kJPx8LUsgQMof70,39767
jinja2/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
jinja2/runtime.py,sha256=wVRlkEmAgNU67AIQDqLvI6UkNLkzDqpLA-z4Mi3vl3g,35054
jinja2/sandbox.py,sha256=-8zxR6TO9kUkciAVFsIKu8Oq-C7PTeYEdZ5TtA55-gw,14600
jinja2/tests.py,sha256=Am5Z6Lmfr2XaH_npIfJJ8MdXtWsbLjMULZJulTAj30E,5905
jinja2/utils.py,sha256=udQxWIKaq4QDCZiXN31ngKOaGGdaMA5fl0JMaM-F6fg,26971
jinja2/visitor.py,sha256=ZmeLuTj66ic35-uFH-1m0EKXiw4ObDDb_WuE6h5vPFg,3572

View File

@ -0,0 +1,5 @@
Wheel-Version: 1.0
Generator: bdist_wheel (0.32.3)
Root-Is-Purelib: true
Tag: py3-none-any

View File

@ -0,0 +1,3 @@
[babel.extractors]
jinja2 = jinja2.ext:babel_extract [i18n]

View File

@ -0,0 +1,101 @@
Metadata-Version: 2.1
Name: MarkupSafe
Version: 2.0.1
Summary: Safely add untrusted strings to HTML/XML markup.
Home-page: https://palletsprojects.com/p/markupsafe/
Author: Armin Ronacher
Author-email: armin.ronacher@active-4.com
Maintainer: Pallets
Maintainer-email: contact@palletsprojects.com
License: BSD-3-Clause
Project-URL: Donate, https://palletsprojects.com/donate
Project-URL: Documentation, https://markupsafe.palletsprojects.com/
Project-URL: Changes, https://markupsafe.palletsprojects.com/changes/
Project-URL: Source Code, https://github.com/pallets/markupsafe/
Project-URL: Issue Tracker, https://github.com/pallets/markupsafe/issues/
Project-URL: Twitter, https://twitter.com/PalletsTeam
Project-URL: Chat, https://discord.gg/pallets
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Web Environment
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Topic :: Internet :: WWW/HTTP :: Dynamic Content
Classifier: Topic :: Text Processing :: Markup :: HTML
Requires-Python: >=3.6
Description-Content-Type: text/x-rst
License-File: LICENSE.rst
MarkupSafe
==========
MarkupSafe implements a text object that escapes characters so it is
safe to use in HTML and XML. Characters that have special meanings are
replaced so that they display as the actual characters. This mitigates
injection attacks, meaning untrusted user input can safely be displayed
on a page.
Installing
----------
Install and update using `pip`_:
.. code-block:: text
pip install -U MarkupSafe
.. _pip: https://pip.pypa.io/en/stable/quickstart/
Examples
--------
.. code-block:: pycon
>>> from markupsafe import Markup, escape
>>> # escape replaces special characters and wraps in Markup
>>> escape("<script>alert(document.cookie);</script>")
Markup('&lt;script&gt;alert(document.cookie);&lt;/script&gt;')
>>> # wrap in Markup to mark text "safe" and prevent escaping
>>> Markup("<strong>Hello</strong>")
Markup('<strong>hello</strong>')
>>> escape(Markup("<strong>Hello</strong>"))
Markup('<strong>hello</strong>')
>>> # Markup is a str subclass
>>> # methods and operators escape their arguments
>>> template = Markup("Hello <em>{name}</em>")
>>> template.format(name='"World"')
Markup('Hello <em>&#34;World&#34;</em>')
Donate
------
The Pallets organization develops and supports MarkupSafe and other
popular packages. In order to grow the community of contributors and
users, and allow the maintainers to devote more time to the projects,
`please donate today`_.
.. _please donate today: https://palletsprojects.com/donate
Links
-----
- Documentation: https://markupsafe.palletsprojects.com/
- Changes: https://markupsafe.palletsprojects.com/changes/
- PyPI Releases: https://pypi.org/project/MarkupSafe/
- Source Code: https://github.com/pallets/markupsafe/
- Issue Tracker: https://github.com/pallets/markupsafe/issues/
- Website: https://palletsprojects.com/p/markupsafe/
- Twitter: https://twitter.com/PalletsTeam
- Chat: https://discord.gg/pallets

View File

@ -0,0 +1,34 @@
CHANGES.rst
LICENSE.rst
MANIFEST.in
README.rst
setup.cfg
setup.py
tox.ini
docs/Makefile
docs/changes.rst
docs/conf.py
docs/escaping.rst
docs/formatting.rst
docs/html.rst
docs/index.rst
docs/license.rst
docs/make.bat
requirements/dev.txt
requirements/docs.txt
requirements/tests.txt
requirements/typing.txt
src/MarkupSafe.egg-info/PKG-INFO
src/MarkupSafe.egg-info/SOURCES.txt
src/MarkupSafe.egg-info/dependency_links.txt
src/MarkupSafe.egg-info/top_level.txt
src/markupsafe/__init__.py
src/markupsafe/_native.py
src/markupsafe/_speedups.c
src/markupsafe/_speedups.pyi
src/markupsafe/py.typed
tests/conftest.py
tests/test_escape.py
tests/test_exception_custom_html.py
tests/test_leak.py
tests/test_markupsafe.py

View File

@ -0,0 +1,12 @@
../markupsafe/__init__.py
../markupsafe/__pycache__/__init__.cpython-36.pyc
../markupsafe/__pycache__/_native.cpython-36.pyc
../markupsafe/_native.py
../markupsafe/_speedups.c
../markupsafe/_speedups.cpython-36m-arm-linux-gnueabihf.so
../markupsafe/_speedups.pyi
../markupsafe/py.typed
PKG-INFO
SOURCES.txt
dependency_links.txt
top_level.txt

View File

@ -0,0 +1,28 @@
Copyright 2007 Pallets
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

View File

@ -0,0 +1,129 @@
Metadata-Version: 2.1
Name: Werkzeug
Version: 2.0.2
Summary: The comprehensive WSGI web application library.
Home-page: https://palletsprojects.com/p/werkzeug/
Author: Armin Ronacher
Author-email: armin.ronacher@active-4.com
Maintainer: Pallets
Maintainer-email: contact@palletsprojects.com
License: BSD-3-Clause
Project-URL: Donate, https://palletsprojects.com/donate
Project-URL: Documentation, https://werkzeug.palletsprojects.com/
Project-URL: Changes, https://werkzeug.palletsprojects.com/changes/
Project-URL: Source Code, https://github.com/pallets/werkzeug/
Project-URL: Issue Tracker, https://github.com/pallets/werkzeug/issues/
Project-URL: Twitter, https://twitter.com/PalletsTeam
Project-URL: Chat, https://discord.gg/pallets
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Environment :: Web Environment
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Classifier: Topic :: Internet :: WWW/HTTP :: Dynamic Content
Classifier: Topic :: Internet :: WWW/HTTP :: WSGI
Classifier: Topic :: Internet :: WWW/HTTP :: WSGI :: Application
Classifier: Topic :: Internet :: WWW/HTTP :: WSGI :: Middleware
Classifier: Topic :: Software Development :: Libraries :: Application Frameworks
Requires-Python: >=3.6
Description-Content-Type: text/x-rst
License-File: LICENSE.rst
Requires-Dist: dataclasses ; python_version < "3.7"
Provides-Extra: watchdog
Requires-Dist: watchdog ; extra == 'watchdog'
Werkzeug
========
*werkzeug* German noun: "tool". Etymology: *werk* ("work"), *zeug* ("stuff")
Werkzeug is a comprehensive `WSGI`_ web application library. It began as
a simple collection of various utilities for WSGI applications and has
become one of the most advanced WSGI utility libraries.
It includes:
- An interactive debugger that allows inspecting stack traces and
source code in the browser with an interactive interpreter for any
frame in the stack.
- A full-featured request object with objects to interact with
headers, query args, form data, files, and cookies.
- A response object that can wrap other WSGI applications and handle
streaming data.
- A routing system for matching URLs to endpoints and generating URLs
for endpoints, with an extensible system for capturing variables
from URLs.
- HTTP utilities to handle entity tags, cache control, dates, user
agents, cookies, files, and more.
- A threaded WSGI server for use while developing applications
locally.
- A test client for simulating HTTP requests during testing without
requiring running a server.
Werkzeug doesn't enforce any dependencies. It is up to the developer to
choose a template engine, database adapter, and even how to handle
requests. It can be used to build all sorts of end user applications
such as blogs, wikis, or bulletin boards.
`Flask`_ wraps Werkzeug, using it to handle the details of WSGI while
providing more structure and patterns for defining powerful
applications.
.. _WSGI: https://wsgi.readthedocs.io/en/latest/
.. _Flask: https://www.palletsprojects.com/p/flask/
Installing
----------
Install and update using `pip`_:
.. code-block:: text
pip install -U Werkzeug
.. _pip: https://pip.pypa.io/en/stable/getting-started/
A Simple Example
----------------
.. code-block:: python
from werkzeug.wrappers import Request, Response
@Request.application
def application(request):
return Response('Hello, World!')
if __name__ == '__main__':
from werkzeug.serving import run_simple
run_simple('localhost', 4000, application)
Donate
------
The Pallets organization develops and supports Werkzeug and other
popular packages. In order to grow the community of contributors and
users, and allow the maintainers to devote more time to the projects,
`please donate today`_.
.. _please donate today: https://palletsprojects.com/donate
Links
-----
- Documentation: https://werkzeug.palletsprojects.com/
- Changes: https://werkzeug.palletsprojects.com/changes/
- PyPI Releases: https://pypi.org/project/Werkzeug/
- Source Code: https://github.com/pallets/werkzeug/
- Issue Tracker: https://github.com/pallets/werkzeug/issues/
- Website: https://palletsprojects.com/p/werkzeug/
- Twitter: https://twitter.com/PalletsTeam
- Chat: https://discord.gg/pallets

View File

@ -0,0 +1,111 @@
Werkzeug-2.0.2.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
Werkzeug-2.0.2.dist-info/LICENSE.rst,sha256=O0nc7kEF6ze6wQ-vG-JgQI_oXSUrjp3y4JefweCUQ3s,1475
Werkzeug-2.0.2.dist-info/METADATA,sha256=vh_xrARtpmkFYnWRAgfSiHgl66LH143rMfAfPZo-R_E,4452
Werkzeug-2.0.2.dist-info/RECORD,,
Werkzeug-2.0.2.dist-info/WHEEL,sha256=_NOXIqFgOaYmlm9RJLPQZ13BJuEIrp5jx5ptRD5uh3Y,92
Werkzeug-2.0.2.dist-info/top_level.txt,sha256=QRyj2VjwJoQkrwjwFIOlB8Xg3r9un0NtqVHQF-15xaw,9
werkzeug/__init__.py,sha256=Wx1PLCftJ7UAS0fBXEO4Prdr6kvEQ124Stwg-XwyhW4,188
werkzeug/__pycache__/__init__.cpython-36.pyc,,
werkzeug/__pycache__/_internal.cpython-36.pyc,,
werkzeug/__pycache__/_reloader.cpython-36.pyc,,
werkzeug/__pycache__/datastructures.cpython-36.pyc,,
werkzeug/__pycache__/exceptions.cpython-36.pyc,,
werkzeug/__pycache__/filesystem.cpython-36.pyc,,
werkzeug/__pycache__/formparser.cpython-36.pyc,,
werkzeug/__pycache__/http.cpython-36.pyc,,
werkzeug/__pycache__/local.cpython-36.pyc,,
werkzeug/__pycache__/routing.cpython-36.pyc,,
werkzeug/__pycache__/security.cpython-36.pyc,,
werkzeug/__pycache__/serving.cpython-36.pyc,,
werkzeug/__pycache__/test.cpython-36.pyc,,
werkzeug/__pycache__/testapp.cpython-36.pyc,,
werkzeug/__pycache__/urls.cpython-36.pyc,,
werkzeug/__pycache__/user_agent.cpython-36.pyc,,
werkzeug/__pycache__/useragents.cpython-36.pyc,,
werkzeug/__pycache__/utils.cpython-36.pyc,,
werkzeug/__pycache__/wsgi.cpython-36.pyc,,
werkzeug/_internal.py,sha256=_QKkvdaG4pDFwK68c0EpPzYJGe9Y7toRAT1cBbC-CxU,18572
werkzeug/_reloader.py,sha256=B1hEfgsUOz2IginBQM5Zak_eaIF7gr3GS5-0x2OHvAE,13950
werkzeug/datastructures.py,sha256=m79A8rHQEt5B7qVqyrjARXzHL66Katn8S92urGscTw4,97929
werkzeug/datastructures.pyi,sha256=CoVwrQ2Vr9JnbprNL9aE3vOz8mOejT9qysQ-BT53C8Y,34089
werkzeug/debug/__init__.py,sha256=jYA1e1Gw_8EPOytr-BoMdmm0rzP-Z1H0Ih7wIObnKwQ,17968
werkzeug/debug/__pycache__/__init__.cpython-36.pyc,,
werkzeug/debug/__pycache__/console.cpython-36.pyc,,
werkzeug/debug/__pycache__/repr.cpython-36.pyc,,
werkzeug/debug/__pycache__/tbtools.cpython-36.pyc,,
werkzeug/debug/console.py,sha256=E1nBMEvFkX673ShQjPtVY-byYatfX9MN-dBMjRI8a8E,5897
werkzeug/debug/repr.py,sha256=QCSHENKsChEZDCIApkVi_UNjhJ77v8BMXK1OfxO189M,9483
werkzeug/debug/shared/FONT_LICENSE,sha256=LwAVEI1oYnvXiNMT9SnCH_TaLCxCpeHziDrMg0gPkAI,4673
werkzeug/debug/shared/ICON_LICENSE.md,sha256=DhA6Y1gUl5Jwfg0NFN9Rj4VWITt8tUx0IvdGf0ux9-s,222
werkzeug/debug/shared/console.png,sha256=bxax6RXXlvOij_KeqvSNX0ojJf83YbnZ7my-3Gx9w2A,507
werkzeug/debug/shared/debugger.js,sha256=tg42SZs1SVmYWZ-_Fj5ELK5-FLHnGNQrei0K2By8Bw8,10521
werkzeug/debug/shared/less.png,sha256=-4-kNRaXJSONVLahrQKUxMwXGm9R4OnZ9SxDGpHlIR4,191
werkzeug/debug/shared/more.png,sha256=GngN7CioHQoV58rH6ojnkYi8c_qED2Aka5FO5UXrReY,200
werkzeug/debug/shared/source.png,sha256=RoGcBTE4CyCB85GBuDGTFlAnUqxwTBiIfDqW15EpnUQ,818
werkzeug/debug/shared/style.css,sha256=h1ZSUVaKNpfbfcYzRb513WAhPySGDQom1uih3uEDxPw,6704
werkzeug/debug/shared/ubuntu.ttf,sha256=1eaHFyepmy4FyDvjLVzpITrGEBu_CZYY94jE0nED1c0,70220
werkzeug/debug/tbtools.py,sha256=AFRrjLDCAps7G5K2-RxNZpXXaEoeFHm68T00f4vlDYA,19362
werkzeug/exceptions.py,sha256=CUwx0pBiNbk4f9cON17ekgKnmLi6HIVFjUmYZc2x0wM,28681
werkzeug/filesystem.py,sha256=JS2Dv2QF98WILxY4_thHl-WMcUcwluF_4igkDPaP1l4,1956
werkzeug/formparser.py,sha256=X-p3Ek4ji8XrKrbmaWxr8StLSc6iuksbpIeweaabs4s,17400
werkzeug/http.py,sha256=oUCXFFMnkOQ-cHbUY_aiqitshcrSzNDq3fEMf1VI_yk,45141
werkzeug/local.py,sha256=bwL-y3-qOZAspJ66W1P36SUApLXJy3UY8nLYbM9kfmY,23183
werkzeug/middleware/__init__.py,sha256=qfqgdT5npwG9ses3-FXQJf3aB95JYP1zchetH_T3PUw,500
werkzeug/middleware/__pycache__/__init__.cpython-36.pyc,,
werkzeug/middleware/__pycache__/dispatcher.cpython-36.pyc,,
werkzeug/middleware/__pycache__/http_proxy.cpython-36.pyc,,
werkzeug/middleware/__pycache__/lint.cpython-36.pyc,,
werkzeug/middleware/__pycache__/profiler.cpython-36.pyc,,
werkzeug/middleware/__pycache__/proxy_fix.cpython-36.pyc,,
werkzeug/middleware/__pycache__/shared_data.cpython-36.pyc,,
werkzeug/middleware/dispatcher.py,sha256=Fh_w-KyWnTSYF-Lfv5dimQ7THSS7afPAZMmvc4zF1gg,2580
werkzeug/middleware/http_proxy.py,sha256=HE8VyhS7CR-E1O6_9b68huv8FLgGGR1DLYqkS3Xcp3Q,7558
werkzeug/middleware/lint.py,sha256=sAg3GcOhICIkwYX5bJGG8n8iebX0Yipq_UH0HvrBvoU,13964
werkzeug/middleware/profiler.py,sha256=QkXk7cqnaPnF8wQu-5SyPCIOT3_kdABUBorQOghVNOA,4899
werkzeug/middleware/proxy_fix.py,sha256=uRgQ3dEvFV8JxUqajHYYYOPEeA_BFqaa51Yp8VW0uzA,6849
werkzeug/middleware/shared_data.py,sha256=xydEqOhAGg0aQJEllPDVfz2-8jHwWvJpAxfPsfPCu7k,10960
werkzeug/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
werkzeug/routing.py,sha256=oqJ32sWIZtIF6zbqfrnwB1Pbv2ShNwPDJd6FYqxdYVo,84527
werkzeug/sansio/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
werkzeug/sansio/__pycache__/__init__.cpython-36.pyc,,
werkzeug/sansio/__pycache__/multipart.cpython-36.pyc,,
werkzeug/sansio/__pycache__/request.cpython-36.pyc,,
werkzeug/sansio/__pycache__/response.cpython-36.pyc,,
werkzeug/sansio/__pycache__/utils.cpython-36.pyc,,
werkzeug/sansio/multipart.py,sha256=bJMCNC2f5xyAaylahNViJ0JqmV4ThLRbDVGVzKwcqrQ,8751
werkzeug/sansio/request.py,sha256=aA9rABkWiG4MhYMByanst2NXkEclsq8SIxhb0LQf0e0,20228
werkzeug/sansio/response.py,sha256=zvCq9HSBBZGBd5Gg412BY9RZIwnKsJl5Kzfd3Kl9sSo,26098
werkzeug/sansio/utils.py,sha256=V5v-UUnX8pm4RehP9Tt_NiUSOJGJGUvKjlW0eOIQldM,4164
werkzeug/security.py,sha256=gPDRuCjkjWrcqj99tBMq8_nHFZLFQjgoW5Ga5XIw9jo,8158
werkzeug/serving.py,sha256=AfgLn0yKr9qXknmwO-0KXJ055oloS4h5DIFDHEu8iHA,38088
werkzeug/test.py,sha256=8gE1l-Y9yAh2i3SI0kgpxIaI4oYZuehIkxxyDFcz6J0,48123
werkzeug/testapp.py,sha256=f48prWSGJhbSrvYb8e1fnAah4BkrLb0enHSdChgsjBY,9471
werkzeug/urls.py,sha256=Du2lreBHvgBh5c2_bcx72g3hzV2ZabXYZsp-picUIJs,41023
werkzeug/user_agent.py,sha256=WclZhpvgLurMF45hsioSbS75H1Zb4iMQGKN3_yZ2oKo,1420
werkzeug/useragents.py,sha256=G8tmv_6vxJaPrLQH3eODNgIYe0_V6KETROQlJI-WxDE,7264
werkzeug/utils.py,sha256=D_dnCLUfodQ4k0GRSpnI6qDoVoaX7-Dza57bx7sabG0,37101
werkzeug/wrappers/__init__.py,sha256=-s75nPbyXHzU_rwmLPDhoMuGbEUk0jZT_n0ZQAOFGf8,654
werkzeug/wrappers/__pycache__/__init__.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/accept.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/auth.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/base_request.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/base_response.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/common_descriptors.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/cors.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/etag.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/json.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/request.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/response.cpython-36.pyc,,
werkzeug/wrappers/__pycache__/user_agent.cpython-36.pyc,,
werkzeug/wrappers/accept.py,sha256=_oZtAQkahvsrPRkNj2fieg7_St9P0NFC3SgZbJKS6xU,429
werkzeug/wrappers/auth.py,sha256=rZPCzGxHk9R55PRkmS90kRywUVjjuMWzCGtH68qCq8U,856
werkzeug/wrappers/base_request.py,sha256=saz9RyNQkvI_XLPYVm29KijNHmD1YzgxDqa0qHTbgss,1174
werkzeug/wrappers/base_response.py,sha256=q_-TaYywT5G4zA-DWDRDJhJSat2_4O7gOPob6ye4_9A,1186
werkzeug/wrappers/common_descriptors.py,sha256=v_kWLH3mvCiSRVJ1FNw7nO3w2UJfzY57UKKB5J4zCvE,898
werkzeug/wrappers/cors.py,sha256=c5UndlZsZvYkbPrp6Gj5iSXxw_VOJDJHskO6-jRmNyQ,846
werkzeug/wrappers/etag.py,sha256=XHWQQs7Mdd1oWezgBIsl-bYe8ydKkRZVil2Qd01D0Mo,846
werkzeug/wrappers/json.py,sha256=HM1btPseGeXca0vnwQN_MvZl6h-qNsFY5YBKXKXFwus,410
werkzeug/wrappers/request.py,sha256=yZGplfC3UqNuykwLJmgywiMhmnoKEGHJOZn_A_ublcQ,24822
werkzeug/wrappers/response.py,sha256=0n8OcQptiM2e550SALLeg7vC1uWsUbCeE1rPZFfXR78,35177
werkzeug/wrappers/user_agent.py,sha256=Wl1-A0-1r8o7cHIZQTB55O4Ged6LpCKENaQDlOY5pXA,435
werkzeug/wsgi.py,sha256=L7s5-Rlt7BRVEZ1m81MaenGfMDP7yL3p1Kxt9Yssqzg,33727

View File

@ -0,0 +1,5 @@
Wheel-Version: 1.0
Generator: bdist_wheel (0.32.3)
Root-Is-Purelib: true
Tag: py3-none-any

View File

@ -0,0 +1,132 @@
import sys
import os
import re
import importlib
import warnings
is_pypy = '__pypy__' in sys.builtin_module_names
warnings.filterwarnings('ignore',
r'.+ distutils\b.+ deprecated',
DeprecationWarning)
def warn_distutils_present():
if 'distutils' not in sys.modules:
return
if is_pypy and sys.version_info < (3, 7):
# PyPy for 3.6 unconditionally imports distutils, so bypass the warning
# https://foss.heptapod.net/pypy/pypy/-/blob/be829135bc0d758997b3566062999ee8b23872b4/lib-python/3/site.py#L250
return
warnings.warn(
"Distutils was imported before Setuptools, but importing Setuptools "
"also replaces the `distutils` module in `sys.modules`. This may lead "
"to undesirable behaviors or errors. To avoid these issues, avoid "
"using distutils directly, ensure that setuptools is installed in the "
"traditional way (e.g. not an editable install), and/or make sure "
"that setuptools is always imported before distutils.")
def clear_distutils():
if 'distutils' not in sys.modules:
return
warnings.warn("Setuptools is replacing distutils.")
mods = [name for name in sys.modules if re.match(r'distutils\b', name)]
for name in mods:
del sys.modules[name]
def enabled():
"""
Allow selection of distutils by environment variable.
"""
which = os.environ.get('SETUPTOOLS_USE_DISTUTILS', 'stdlib')
return which == 'local'
def ensure_local_distutils():
clear_distutils()
# With the DistutilsMetaFinder in place,
# perform an import to cause distutils to be
# loaded from setuptools._distutils. Ref #2906.
add_shim()
importlib.import_module('distutils')
remove_shim()
# check that submodules load as expected
core = importlib.import_module('distutils.core')
assert '_distutils' in core.__file__, core.__file__
def do_override():
"""
Ensure that the local copy of distutils is preferred over stdlib.
See https://github.com/pypa/setuptools/issues/417#issuecomment-392298401
for more motivation.
"""
if enabled():
warn_distutils_present()
ensure_local_distutils()
class DistutilsMetaFinder:
def find_spec(self, fullname, path, target=None):
if path is not None:
return
method_name = 'spec_for_{fullname}'.format(**locals())
method = getattr(self, method_name, lambda: None)
return method()
def spec_for_distutils(self):
import importlib.abc
import importlib.util
class DistutilsLoader(importlib.abc.Loader):
def create_module(self, spec):
return importlib.import_module('setuptools._distutils')
def exec_module(self, module):
pass
return importlib.util.spec_from_loader('distutils', DistutilsLoader())
def spec_for_pip(self):
"""
Ensure stdlib distutils when running under pip.
See pypa/pip#8761 for rationale.
"""
if self.pip_imported_during_build():
return
clear_distutils()
self.spec_for_distutils = lambda: None
@staticmethod
def pip_imported_during_build():
"""
Detect if pip is being imported in a build script. Ref #2355.
"""
import traceback
return any(
frame.f_globals['__file__'].endswith('setup.py')
for frame, line in traceback.walk_stack(None)
)
DISTUTILS_FINDER = DistutilsMetaFinder()
def add_shim():
sys.meta_path.insert(0, DISTUTILS_FINDER)
def remove_shim():
try:
sys.meta_path.remove(DISTUTILS_FINDER)
except ValueError:
pass

View File

@ -0,0 +1 @@
__import__('_distutils_hack').do_override()

View File

@ -0,0 +1,28 @@
Copyright 2014 Pallets
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

View File

@ -0,0 +1,111 @@
Metadata-Version: 2.1
Name: click
Version: 8.0.3
Summary: Composable command line interface toolkit
Home-page: https://palletsprojects.com/p/click/
Author: Armin Ronacher
Author-email: armin.ronacher@active-4.com
Maintainer: Pallets
Maintainer-email: contact@palletsprojects.com
License: BSD-3-Clause
Project-URL: Donate, https://palletsprojects.com/donate
Project-URL: Documentation, https://click.palletsprojects.com/
Project-URL: Changes, https://click.palletsprojects.com/changes/
Project-URL: Source Code, https://github.com/pallets/click/
Project-URL: Issue Tracker, https://github.com/pallets/click/issues/
Project-URL: Twitter, https://twitter.com/PalletsTeam
Project-URL: Chat, https://discord.gg/pallets
Platform: UNKNOWN
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: OS Independent
Classifier: Programming Language :: Python
Requires-Python: >=3.6
Description-Content-Type: text/x-rst
License-File: LICENSE.rst
Requires-Dist: colorama ; platform_system == "Windows"
Requires-Dist: importlib-metadata ; python_version < "3.8"
\$ click\_
==========
Click is a Python package for creating beautiful command line interfaces
in a composable way with as little code as necessary. It's the "Command
Line Interface Creation Kit". It's highly configurable but comes with
sensible defaults out of the box.
It aims to make the process of writing command line tools quick and fun
while also preventing any frustration caused by the inability to
implement an intended CLI API.
Click in three points:
- Arbitrary nesting of commands
- Automatic help page generation
- Supports lazy loading of subcommands at runtime
Installing
----------
Install and update using `pip`_:
.. code-block:: text
$ pip install -U click
.. _pip: https://pip.pypa.io/en/stable/getting-started/
A Simple Example
----------------
.. code-block:: python
import click
@click.command()
@click.option("--count", default=1, help="Number of greetings.")
@click.option("--name", prompt="Your name", help="The person to greet.")
def hello(count, name):
"""Simple program that greets NAME for a total of COUNT times."""
for _ in range(count):
click.echo(f"Hello, {name}!")
if __name__ == '__main__':
hello()
.. code-block:: text
$ python hello.py --count=3
Your name: Click
Hello, Click!
Hello, Click!
Hello, Click!
Donate
------
The Pallets organization develops and supports Click and other popular
packages. In order to grow the community of contributors and users, and
allow the maintainers to devote more time to the projects, `please
donate today`_.
.. _please donate today: https://palletsprojects.com/donate
Links
-----
- Documentation: https://click.palletsprojects.com/
- Changes: https://click.palletsprojects.com/changes/
- PyPI Releases: https://pypi.org/project/click/
- Source Code: https://github.com/pallets/click
- Issue Tracker: https://github.com/pallets/click/issues
- Website: https://palletsprojects.com/p/click
- Twitter: https://twitter.com/PalletsTeam
- Chat: https://discord.gg/pallets

View File

@ -0,0 +1,41 @@
click-8.0.3.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
click-8.0.3.dist-info/LICENSE.rst,sha256=morRBqOU6FO_4h9C9OctWSgZoigF2ZG18ydQKSkrZY0,1475
click-8.0.3.dist-info/METADATA,sha256=_0jCOf3DdGPvKUZUlBukeb1t6Pnxmm_OMGpaBoDthfc,3247
click-8.0.3.dist-info/RECORD,,
click-8.0.3.dist-info/WHEEL,sha256=_NOXIqFgOaYmlm9RJLPQZ13BJuEIrp5jx5ptRD5uh3Y,92
click-8.0.3.dist-info/top_level.txt,sha256=J1ZQogalYS4pphY_lPECoNMfw0HzTSrZglC4Yfwo4xA,6
click/__init__.py,sha256=YkIrDg7-0g5aBS6D2pDe58j3MOaFylHED2_8OXh2fnM,3243
click/__pycache__/__init__.cpython-36.pyc,,
click/__pycache__/_compat.cpython-36.pyc,,
click/__pycache__/_termui_impl.cpython-36.pyc,,
click/__pycache__/_textwrap.cpython-36.pyc,,
click/__pycache__/_unicodefun.cpython-36.pyc,,
click/__pycache__/_winconsole.cpython-36.pyc,,
click/__pycache__/core.cpython-36.pyc,,
click/__pycache__/decorators.cpython-36.pyc,,
click/__pycache__/exceptions.cpython-36.pyc,,
click/__pycache__/formatting.cpython-36.pyc,,
click/__pycache__/globals.cpython-36.pyc,,
click/__pycache__/parser.cpython-36.pyc,,
click/__pycache__/shell_completion.cpython-36.pyc,,
click/__pycache__/termui.cpython-36.pyc,,
click/__pycache__/testing.cpython-36.pyc,,
click/__pycache__/types.cpython-36.pyc,,
click/__pycache__/utils.cpython-36.pyc,,
click/_compat.py,sha256=P15KQumAZC2F2MFe_JSRbvVOJcNosQfMDrdZq0ReCLM,18814
click/_termui_impl.py,sha256=z78J5HF_RTsOBhjNLjoigaqRap3P2pWwEDDAjoZzgUg,23452
click/_textwrap.py,sha256=10fQ64OcBUMuK7mFvh8363_uoOxPlRItZBmKzRJDgoY,1353
click/_unicodefun.py,sha256=JKSh1oSwG_zbjAu4TBCa9tQde2P9FiYcf4MBfy5NdT8,3201
click/_winconsole.py,sha256=5ju3jQkcZD0W27WEMGqmEP4y_crUVzPCqsX_FYb7BO0,7860
click/core.py,sha256=k4PA2z0BT_dmed9I52Q2VLi8r6ekTMCtCQzw2y915Xs,111478
click/decorators.py,sha256=sGkXJGmP7eLtjtmPl_Un2uBTlrhK8s2L22n-yBiDwTw,14864
click/exceptions.py,sha256=7gDaLGuFZBeCNwY9ERMsF2-Z3R9Fvq09Zc6IZSKjseo,9167
click/formatting.py,sha256=Frf0-5W33-loyY_i9qrwXR8-STnW3m5gvyxLVUdyxyk,9706
click/globals.py,sha256=kGPzxq55Ug4dFUrgRV-5oHVPOPdLCUhmYolbrrVBo8g,1985
click/parser.py,sha256=cAEt1uQR8gq3-S9ysqbVU-fdAZNvilxw4ReJ_T1OQMk,19044
click/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
click/shell_completion.py,sha256=_hPI12T9Ex-y5a3WunWnlH0Gca2_urzXFXkDnt7G6Ow,18001
click/termui.py,sha256=Rp2gFE8x7j8sEIoFMOcPmuqxJQVWWTrwEzyC14-sPAw,29006
click/testing.py,sha256=kLR5Qcny1OlgxaGB3gweTr0gQe1yVlmgQRn2esA2Fz4,16020
click/types.py,sha256=VoNZnIlRBAtRRgzavdqVnyfzY5y4U4qzVGI1UvvX1ls,35391
click/utils.py,sha256=avYwX-3l2KkdJNUo8NmncZSoAdEmniQ_M5sdsSYloJ4,18759

View File

@ -0,0 +1,5 @@
Wheel-Version: 1.0
Generator: bdist_wheel (0.32.3)
Root-Is-Purelib: true
Tag: py3-none-any

View File

@ -0,0 +1,75 @@
"""
Click is a simple Python module inspired by the stdlib optparse to make
writing command line scripts fun. Unlike other modules, it's based
around a simple API that does not come with too much magic and is
composable.
"""
from .core import Argument as Argument
from .core import BaseCommand as BaseCommand
from .core import Command as Command
from .core import CommandCollection as CommandCollection
from .core import Context as Context
from .core import Group as Group
from .core import MultiCommand as MultiCommand
from .core import Option as Option
from .core import Parameter as Parameter
from .decorators import argument as argument
from .decorators import command as command
from .decorators import confirmation_option as confirmation_option
from .decorators import group as group
from .decorators import help_option as help_option
from .decorators import make_pass_decorator as make_pass_decorator
from .decorators import option as option
from .decorators import pass_context as pass_context
from .decorators import pass_obj as pass_obj
from .decorators import password_option as password_option
from .decorators import version_option as version_option
from .exceptions import Abort as Abort
from .exceptions import BadArgumentUsage as BadArgumentUsage
from .exceptions import BadOptionUsage as BadOptionUsage
from .exceptions import BadParameter as BadParameter
from .exceptions import ClickException as ClickException
from .exceptions import FileError as FileError
from .exceptions import MissingParameter as MissingParameter
from .exceptions import NoSuchOption as NoSuchOption
from .exceptions import UsageError as UsageError
from .formatting import HelpFormatter as HelpFormatter
from .formatting import wrap_text as wrap_text
from .globals import get_current_context as get_current_context
from .parser import OptionParser as OptionParser
from .termui import clear as clear
from .termui import confirm as confirm
from .termui import echo_via_pager as echo_via_pager
from .termui import edit as edit
from .termui import get_terminal_size as get_terminal_size
from .termui import getchar as getchar
from .termui import launch as launch
from .termui import pause as pause
from .termui import progressbar as progressbar
from .termui import prompt as prompt
from .termui import secho as secho
from .termui import style as style
from .termui import unstyle as unstyle
from .types import BOOL as BOOL
from .types import Choice as Choice
from .types import DateTime as DateTime
from .types import File as File
from .types import FLOAT as FLOAT
from .types import FloatRange as FloatRange
from .types import INT as INT
from .types import IntRange as IntRange
from .types import ParamType as ParamType
from .types import Path as Path
from .types import STRING as STRING
from .types import Tuple as Tuple
from .types import UNPROCESSED as UNPROCESSED
from .types import UUID as UUID
from .utils import echo as echo
from .utils import format_filename as format_filename
from .utils import get_app_dir as get_app_dir
from .utils import get_binary_stream as get_binary_stream
from .utils import get_os_args as get_os_args
from .utils import get_text_stream as get_text_stream
from .utils import open_file as open_file
__version__ = "8.0.3"

View File

@ -0,0 +1,627 @@
import codecs
import io
import os
import re
import sys
import typing as t
from weakref import WeakKeyDictionary
CYGWIN = sys.platform.startswith("cygwin")
MSYS2 = sys.platform.startswith("win") and ("GCC" in sys.version)
# Determine local App Engine environment, per Google's own suggestion
APP_ENGINE = "APPENGINE_RUNTIME" in os.environ and "Development/" in os.environ.get(
"SERVER_SOFTWARE", ""
)
WIN = sys.platform.startswith("win") and not APP_ENGINE and not MSYS2
auto_wrap_for_ansi: t.Optional[t.Callable[[t.TextIO], t.TextIO]] = None
_ansi_re = re.compile(r"\033\[[;?0-9]*[a-zA-Z]")
def get_filesystem_encoding() -> str:
return sys.getfilesystemencoding() or sys.getdefaultencoding()
def _make_text_stream(
stream: t.BinaryIO,
encoding: t.Optional[str],
errors: t.Optional[str],
force_readable: bool = False,
force_writable: bool = False,
) -> t.TextIO:
if encoding is None:
encoding = get_best_encoding(stream)
if errors is None:
errors = "replace"
return _NonClosingTextIOWrapper(
stream,
encoding,
errors,
line_buffering=True,
force_readable=force_readable,
force_writable=force_writable,
)
def is_ascii_encoding(encoding: str) -> bool:
"""Checks if a given encoding is ascii."""
try:
return codecs.lookup(encoding).name == "ascii"
except LookupError:
return False
def get_best_encoding(stream: t.IO) -> str:
"""Returns the default stream encoding if not found."""
rv = getattr(stream, "encoding", None) or sys.getdefaultencoding()
if is_ascii_encoding(rv):
return "utf-8"
return rv
class _NonClosingTextIOWrapper(io.TextIOWrapper):
def __init__(
self,
stream: t.BinaryIO,
encoding: t.Optional[str],
errors: t.Optional[str],
force_readable: bool = False,
force_writable: bool = False,
**extra: t.Any,
) -> None:
self._stream = stream = t.cast(
t.BinaryIO, _FixupStream(stream, force_readable, force_writable)
)
super().__init__(stream, encoding, errors, **extra)
def __del__(self) -> None:
try:
self.detach()
except Exception:
pass
def isatty(self) -> bool:
# https://bitbucket.org/pypy/pypy/issue/1803
return self._stream.isatty()
class _FixupStream:
"""The new io interface needs more from streams than streams
traditionally implement. As such, this fix-up code is necessary in
some circumstances.
The forcing of readable and writable flags are there because some tools
put badly patched objects on sys (one such offender are certain version
of jupyter notebook).
"""
def __init__(
self,
stream: t.BinaryIO,
force_readable: bool = False,
force_writable: bool = False,
):
self._stream = stream
self._force_readable = force_readable
self._force_writable = force_writable
def __getattr__(self, name: str) -> t.Any:
return getattr(self._stream, name)
def read1(self, size: int) -> bytes:
f = getattr(self._stream, "read1", None)
if f is not None:
return t.cast(bytes, f(size))
return self._stream.read(size)
def readable(self) -> bool:
if self._force_readable:
return True
x = getattr(self._stream, "readable", None)
if x is not None:
return t.cast(bool, x())
try:
self._stream.read(0)
except Exception:
return False
return True
def writable(self) -> bool:
if self._force_writable:
return True
x = getattr(self._stream, "writable", None)
if x is not None:
return t.cast(bool, x())
try:
self._stream.write("") # type: ignore
except Exception:
try:
self._stream.write(b"")
except Exception:
return False
return True
def seekable(self) -> bool:
x = getattr(self._stream, "seekable", None)
if x is not None:
return t.cast(bool, x())
try:
self._stream.seek(self._stream.tell())
except Exception:
return False
return True
def _is_binary_reader(stream: t.IO, default: bool = False) -> bool:
try:
return isinstance(stream.read(0), bytes)
except Exception:
return default
# This happens in some cases where the stream was already
# closed. In this case, we assume the default.
def _is_binary_writer(stream: t.IO, default: bool = False) -> bool:
try:
stream.write(b"")
except Exception:
try:
stream.write("")
return False
except Exception:
pass
return default
return True
def _find_binary_reader(stream: t.IO) -> t.Optional[t.BinaryIO]:
# We need to figure out if the given stream is already binary.
# This can happen because the official docs recommend detaching
# the streams to get binary streams. Some code might do this, so
# we need to deal with this case explicitly.
if _is_binary_reader(stream, False):
return t.cast(t.BinaryIO, stream)
buf = getattr(stream, "buffer", None)
# Same situation here; this time we assume that the buffer is
# actually binary in case it's closed.
if buf is not None and _is_binary_reader(buf, True):
return t.cast(t.BinaryIO, buf)
return None
def _find_binary_writer(stream: t.IO) -> t.Optional[t.BinaryIO]:
# We need to figure out if the given stream is already binary.
# This can happen because the official docs recommend detaching
# the streams to get binary streams. Some code might do this, so
# we need to deal with this case explicitly.
if _is_binary_writer(stream, False):
return t.cast(t.BinaryIO, stream)
buf = getattr(stream, "buffer", None)
# Same situation here; this time we assume that the buffer is
# actually binary in case it's closed.
if buf is not None and _is_binary_writer(buf, True):
return t.cast(t.BinaryIO, buf)
return None
def _stream_is_misconfigured(stream: t.TextIO) -> bool:
"""A stream is misconfigured if its encoding is ASCII."""
# If the stream does not have an encoding set, we assume it's set
# to ASCII. This appears to happen in certain unittest
# environments. It's not quite clear what the correct behavior is
# but this at least will force Click to recover somehow.
return is_ascii_encoding(getattr(stream, "encoding", None) or "ascii")
def _is_compat_stream_attr(stream: t.TextIO, attr: str, value: t.Optional[str]) -> bool:
"""A stream attribute is compatible if it is equal to the
desired value or the desired value is unset and the attribute
has a value.
"""
stream_value = getattr(stream, attr, None)
return stream_value == value or (value is None and stream_value is not None)
def _is_compatible_text_stream(
stream: t.TextIO, encoding: t.Optional[str], errors: t.Optional[str]
) -> bool:
"""Check if a stream's encoding and errors attributes are
compatible with the desired values.
"""
return _is_compat_stream_attr(
stream, "encoding", encoding
) and _is_compat_stream_attr(stream, "errors", errors)
def _force_correct_text_stream(
text_stream: t.IO,
encoding: t.Optional[str],
errors: t.Optional[str],
is_binary: t.Callable[[t.IO, bool], bool],
find_binary: t.Callable[[t.IO], t.Optional[t.BinaryIO]],
force_readable: bool = False,
force_writable: bool = False,
) -> t.TextIO:
if is_binary(text_stream, False):
binary_reader = t.cast(t.BinaryIO, text_stream)
else:
text_stream = t.cast(t.TextIO, text_stream)
# If the stream looks compatible, and won't default to a
# misconfigured ascii encoding, return it as-is.
if _is_compatible_text_stream(text_stream, encoding, errors) and not (
encoding is None and _stream_is_misconfigured(text_stream)
):
return text_stream
# Otherwise, get the underlying binary reader.
possible_binary_reader = find_binary(text_stream)
# If that's not possible, silently use the original reader
# and get mojibake instead of exceptions.
if possible_binary_reader is None:
return text_stream
binary_reader = possible_binary_reader
# Default errors to replace instead of strict in order to get
# something that works.
if errors is None:
errors = "replace"
# Wrap the binary stream in a text stream with the correct
# encoding parameters.
return _make_text_stream(
binary_reader,
encoding,
errors,
force_readable=force_readable,
force_writable=force_writable,
)
def _force_correct_text_reader(
text_reader: t.IO,
encoding: t.Optional[str],
errors: t.Optional[str],
force_readable: bool = False,
) -> t.TextIO:
return _force_correct_text_stream(
text_reader,
encoding,
errors,
_is_binary_reader,
_find_binary_reader,
force_readable=force_readable,
)
def _force_correct_text_writer(
text_writer: t.IO,
encoding: t.Optional[str],
errors: t.Optional[str],
force_writable: bool = False,
) -> t.TextIO:
return _force_correct_text_stream(
text_writer,
encoding,
errors,
_is_binary_writer,
_find_binary_writer,
force_writable=force_writable,
)
def get_binary_stdin() -> t.BinaryIO:
reader = _find_binary_reader(sys.stdin)
if reader is None:
raise RuntimeError("Was not able to determine binary stream for sys.stdin.")
return reader
def get_binary_stdout() -> t.BinaryIO:
writer = _find_binary_writer(sys.stdout)
if writer is None:
raise RuntimeError("Was not able to determine binary stream for sys.stdout.")
return writer
def get_binary_stderr() -> t.BinaryIO:
writer = _find_binary_writer(sys.stderr)
if writer is None:
raise RuntimeError("Was not able to determine binary stream for sys.stderr.")
return writer
def get_text_stdin(
encoding: t.Optional[str] = None, errors: t.Optional[str] = None
) -> t.TextIO:
rv = _get_windows_console_stream(sys.stdin, encoding, errors)
if rv is not None:
return rv
return _force_correct_text_reader(sys.stdin, encoding, errors, force_readable=True)
def get_text_stdout(
encoding: t.Optional[str] = None, errors: t.Optional[str] = None
) -> t.TextIO:
rv = _get_windows_console_stream(sys.stdout, encoding, errors)
if rv is not None:
return rv
return _force_correct_text_writer(sys.stdout, encoding, errors, force_writable=True)
def get_text_stderr(
encoding: t.Optional[str] = None, errors: t.Optional[str] = None
) -> t.TextIO:
rv = _get_windows_console_stream(sys.stderr, encoding, errors)
if rv is not None:
return rv
return _force_correct_text_writer(sys.stderr, encoding, errors, force_writable=True)
def _wrap_io_open(
file: t.Union[str, os.PathLike, int],
mode: str,
encoding: t.Optional[str],
errors: t.Optional[str],
) -> t.IO:
"""Handles not passing ``encoding`` and ``errors`` in binary mode."""
if "b" in mode:
return open(file, mode)
return open(file, mode, encoding=encoding, errors=errors)
def open_stream(
filename: str,
mode: str = "r",
encoding: t.Optional[str] = None,
errors: t.Optional[str] = "strict",
atomic: bool = False,
) -> t.Tuple[t.IO, bool]:
binary = "b" in mode
# Standard streams first. These are simple because they don't need
# special handling for the atomic flag. It's entirely ignored.
if filename == "-":
if any(m in mode for m in ["w", "a", "x"]):
if binary:
return get_binary_stdout(), False
return get_text_stdout(encoding=encoding, errors=errors), False
if binary:
return get_binary_stdin(), False
return get_text_stdin(encoding=encoding, errors=errors), False
# Non-atomic writes directly go out through the regular open functions.
if not atomic:
return _wrap_io_open(filename, mode, encoding, errors), True
# Some usability stuff for atomic writes
if "a" in mode:
raise ValueError(
"Appending to an existing file is not supported, because that"
" would involve an expensive `copy`-operation to a temporary"
" file. Open the file in normal `w`-mode and copy explicitly"
" if that's what you're after."
)
if "x" in mode:
raise ValueError("Use the `overwrite`-parameter instead.")
if "w" not in mode:
raise ValueError("Atomic writes only make sense with `w`-mode.")
# Atomic writes are more complicated. They work by opening a file
# as a proxy in the same folder and then using the fdopen
# functionality to wrap it in a Python file. Then we wrap it in an
# atomic file that moves the file over on close.
import errno
import random
try:
perm: t.Optional[int] = os.stat(filename).st_mode
except OSError:
perm = None
flags = os.O_RDWR | os.O_CREAT | os.O_EXCL
if binary:
flags |= getattr(os, "O_BINARY", 0)
while True:
tmp_filename = os.path.join(
os.path.dirname(filename),
f".__atomic-write{random.randrange(1 << 32):08x}",
)
try:
fd = os.open(tmp_filename, flags, 0o666 if perm is None else perm)
break
except OSError as e:
if e.errno == errno.EEXIST or (
os.name == "nt"
and e.errno == errno.EACCES
and os.path.isdir(e.filename)
and os.access(e.filename, os.W_OK)
):
continue
raise
if perm is not None:
os.chmod(tmp_filename, perm) # in case perm includes bits in umask
f = _wrap_io_open(fd, mode, encoding, errors)
af = _AtomicFile(f, tmp_filename, os.path.realpath(filename))
return t.cast(t.IO, af), True
class _AtomicFile:
def __init__(self, f: t.IO, tmp_filename: str, real_filename: str) -> None:
self._f = f
self._tmp_filename = tmp_filename
self._real_filename = real_filename
self.closed = False
@property
def name(self) -> str:
return self._real_filename
def close(self, delete: bool = False) -> None:
if self.closed:
return
self._f.close()
os.replace(self._tmp_filename, self._real_filename)
self.closed = True
def __getattr__(self, name: str) -> t.Any:
return getattr(self._f, name)
def __enter__(self) -> "_AtomicFile":
return self
def __exit__(self, exc_type, exc_value, tb): # type: ignore
self.close(delete=exc_type is not None)
def __repr__(self) -> str:
return repr(self._f)
def strip_ansi(value: str) -> str:
return _ansi_re.sub("", value)
def _is_jupyter_kernel_output(stream: t.IO) -> bool:
while isinstance(stream, (_FixupStream, _NonClosingTextIOWrapper)):
stream = stream._stream
return stream.__class__.__module__.startswith("ipykernel.")
def should_strip_ansi(
stream: t.Optional[t.IO] = None, color: t.Optional[bool] = None
) -> bool:
if color is None:
if stream is None:
stream = sys.stdin
return not isatty(stream) and not _is_jupyter_kernel_output(stream)
return not color
# On Windows, wrap the output streams with colorama to support ANSI
# color codes.
# NOTE: double check is needed so mypy does not analyze this on Linux
if sys.platform.startswith("win") and WIN:
from ._winconsole import _get_windows_console_stream
def _get_argv_encoding() -> str:
import locale
return locale.getpreferredencoding()
_ansi_stream_wrappers: t.MutableMapping[t.TextIO, t.TextIO] = WeakKeyDictionary()
def auto_wrap_for_ansi(
stream: t.TextIO, color: t.Optional[bool] = None
) -> t.TextIO:
"""Support ANSI color and style codes on Windows by wrapping a
stream with colorama.
"""
try:
cached = _ansi_stream_wrappers.get(stream)
except Exception:
cached = None
if cached is not None:
return cached
import colorama
strip = should_strip_ansi(stream, color)
ansi_wrapper = colorama.AnsiToWin32(stream, strip=strip)
rv = t.cast(t.TextIO, ansi_wrapper.stream)
_write = rv.write
def _safe_write(s):
try:
return _write(s)
except BaseException:
ansi_wrapper.reset_all()
raise
rv.write = _safe_write
try:
_ansi_stream_wrappers[stream] = rv
except Exception:
pass
return rv
else:
def _get_argv_encoding() -> str:
return getattr(sys.stdin, "encoding", None) or get_filesystem_encoding()
def _get_windows_console_stream(
f: t.TextIO, encoding: t.Optional[str], errors: t.Optional[str]
) -> t.Optional[t.TextIO]:
return None
def term_len(x: str) -> int:
return len(strip_ansi(x))
def isatty(stream: t.IO) -> bool:
try:
return stream.isatty()
except Exception:
return False
def _make_cached_stream_func(
src_func: t.Callable[[], t.TextIO], wrapper_func: t.Callable[[], t.TextIO]
) -> t.Callable[[], t.TextIO]:
cache: t.MutableMapping[t.TextIO, t.TextIO] = WeakKeyDictionary()
def func() -> t.TextIO:
stream = src_func()
try:
rv = cache.get(stream)
except Exception:
rv = None
if rv is not None:
return rv
rv = wrapper_func()
try:
cache[stream] = rv
except Exception:
pass
return rv
return func
_default_text_stdin = _make_cached_stream_func(lambda: sys.stdin, get_text_stdin)
_default_text_stdout = _make_cached_stream_func(lambda: sys.stdout, get_text_stdout)
_default_text_stderr = _make_cached_stream_func(lambda: sys.stderr, get_text_stderr)
binary_streams: t.Mapping[str, t.Callable[[], t.BinaryIO]] = {
"stdin": get_binary_stdin,
"stdout": get_binary_stdout,
"stderr": get_binary_stderr,
}
text_streams: t.Mapping[
str, t.Callable[[t.Optional[str], t.Optional[str]], t.TextIO]
] = {
"stdin": get_text_stdin,
"stdout": get_text_stdout,
"stderr": get_text_stderr,
}

View File

@ -0,0 +1,718 @@
"""
This module contains implementations for the termui module. To keep the
import time of Click down, some infrequently used functionality is
placed in this module and only imported as needed.
"""
import contextlib
import math
import os
import sys
import time
import typing as t
from gettext import gettext as _
from ._compat import _default_text_stdout
from ._compat import CYGWIN
from ._compat import get_best_encoding
from ._compat import isatty
from ._compat import open_stream
from ._compat import strip_ansi
from ._compat import term_len
from ._compat import WIN
from .exceptions import ClickException
from .utils import echo
V = t.TypeVar("V")
if os.name == "nt":
BEFORE_BAR = "\r"
AFTER_BAR = "\n"
else:
BEFORE_BAR = "\r\033[?25l"
AFTER_BAR = "\033[?25h\n"
class ProgressBar(t.Generic[V]):
def __init__(
self,
iterable: t.Optional[t.Iterable[V]],
length: t.Optional[int] = None,
fill_char: str = "#",
empty_char: str = " ",
bar_template: str = "%(bar)s",
info_sep: str = " ",
show_eta: bool = True,
show_percent: t.Optional[bool] = None,
show_pos: bool = False,
item_show_func: t.Optional[t.Callable[[t.Optional[V]], t.Optional[str]]] = None,
label: t.Optional[str] = None,
file: t.Optional[t.TextIO] = None,
color: t.Optional[bool] = None,
update_min_steps: int = 1,
width: int = 30,
) -> None:
self.fill_char = fill_char
self.empty_char = empty_char
self.bar_template = bar_template
self.info_sep = info_sep
self.show_eta = show_eta
self.show_percent = show_percent
self.show_pos = show_pos
self.item_show_func = item_show_func
self.label = label or ""
if file is None:
file = _default_text_stdout()
self.file = file
self.color = color
self.update_min_steps = update_min_steps
self._completed_intervals = 0
self.width = width
self.autowidth = width == 0
if length is None:
from operator import length_hint
length = length_hint(iterable, -1)
if length == -1:
length = None
if iterable is None:
if length is None:
raise TypeError("iterable or length is required")
iterable = t.cast(t.Iterable[V], range(length))
self.iter = iter(iterable)
self.length = length
self.pos = 0
self.avg: t.List[float] = []
self.start = self.last_eta = time.time()
self.eta_known = False
self.finished = False
self.max_width: t.Optional[int] = None
self.entered = False
self.current_item: t.Optional[V] = None
self.is_hidden = not isatty(self.file)
self._last_line: t.Optional[str] = None
def __enter__(self) -> "ProgressBar":
self.entered = True
self.render_progress()
return self
def __exit__(self, exc_type, exc_value, tb): # type: ignore
self.render_finish()
def __iter__(self) -> t.Iterator[V]:
if not self.entered:
raise RuntimeError("You need to use progress bars in a with block.")
self.render_progress()
return self.generator()
def __next__(self) -> V:
# Iteration is defined in terms of a generator function,
# returned by iter(self); use that to define next(). This works
# because `self.iter` is an iterable consumed by that generator,
# so it is re-entry safe. Calling `next(self.generator())`
# twice works and does "what you want".
return next(iter(self))
def render_finish(self) -> None:
if self.is_hidden:
return
self.file.write(AFTER_BAR)
self.file.flush()
@property
def pct(self) -> float:
if self.finished:
return 1.0
return min(self.pos / (float(self.length or 1) or 1), 1.0)
@property
def time_per_iteration(self) -> float:
if not self.avg:
return 0.0
return sum(self.avg) / float(len(self.avg))
@property
def eta(self) -> float:
if self.length is not None and not self.finished:
return self.time_per_iteration * (self.length - self.pos)
return 0.0
def format_eta(self) -> str:
if self.eta_known:
t = int(self.eta)
seconds = t % 60
t //= 60
minutes = t % 60
t //= 60
hours = t % 24
t //= 24
if t > 0:
return f"{t}d {hours:02}:{minutes:02}:{seconds:02}"
else:
return f"{hours:02}:{minutes:02}:{seconds:02}"
return ""
def format_pos(self) -> str:
pos = str(self.pos)
if self.length is not None:
pos += f"/{self.length}"
return pos
def format_pct(self) -> str:
return f"{int(self.pct * 100): 4}%"[1:]
def format_bar(self) -> str:
if self.length is not None:
bar_length = int(self.pct * self.width)
bar = self.fill_char * bar_length
bar += self.empty_char * (self.width - bar_length)
elif self.finished:
bar = self.fill_char * self.width
else:
chars = list(self.empty_char * (self.width or 1))
if self.time_per_iteration != 0:
chars[
int(
(math.cos(self.pos * self.time_per_iteration) / 2.0 + 0.5)
* self.width
)
] = self.fill_char
bar = "".join(chars)
return bar
def format_progress_line(self) -> str:
show_percent = self.show_percent
info_bits = []
if self.length is not None and show_percent is None:
show_percent = not self.show_pos
if self.show_pos:
info_bits.append(self.format_pos())
if show_percent:
info_bits.append(self.format_pct())
if self.show_eta and self.eta_known and not self.finished:
info_bits.append(self.format_eta())
if self.item_show_func is not None:
item_info = self.item_show_func(self.current_item)
if item_info is not None:
info_bits.append(item_info)
return (
self.bar_template
% {
"label": self.label,
"bar": self.format_bar(),
"info": self.info_sep.join(info_bits),
}
).rstrip()
def render_progress(self) -> None:
import shutil
if self.is_hidden:
# Only output the label as it changes if the output is not a
# TTY. Use file=stderr if you expect to be piping stdout.
if self._last_line != self.label:
self._last_line = self.label
echo(self.label, file=self.file, color=self.color)
return
buf = []
# Update width in case the terminal has been resized
if self.autowidth:
old_width = self.width
self.width = 0
clutter_length = term_len(self.format_progress_line())
new_width = max(0, shutil.get_terminal_size().columns - clutter_length)
if new_width < old_width:
buf.append(BEFORE_BAR)
buf.append(" " * self.max_width) # type: ignore
self.max_width = new_width
self.width = new_width
clear_width = self.width
if self.max_width is not None:
clear_width = self.max_width
buf.append(BEFORE_BAR)
line = self.format_progress_line()
line_len = term_len(line)
if self.max_width is None or self.max_width < line_len:
self.max_width = line_len
buf.append(line)
buf.append(" " * (clear_width - line_len))
line = "".join(buf)
# Render the line only if it changed.
if line != self._last_line:
self._last_line = line
echo(line, file=self.file, color=self.color, nl=False)
self.file.flush()
def make_step(self, n_steps: int) -> None:
self.pos += n_steps
if self.length is not None and self.pos >= self.length:
self.finished = True
if (time.time() - self.last_eta) < 1.0:
return
self.last_eta = time.time()
# self.avg is a rolling list of length <= 7 of steps where steps are
# defined as time elapsed divided by the total progress through
# self.length.
if self.pos:
step = (time.time() - self.start) / self.pos
else:
step = time.time() - self.start
self.avg = self.avg[-6:] + [step]
self.eta_known = self.length is not None
def update(self, n_steps: int, current_item: t.Optional[V] = None) -> None:
"""Update the progress bar by advancing a specified number of
steps, and optionally set the ``current_item`` for this new
position.
:param n_steps: Number of steps to advance.
:param current_item: Optional item to set as ``current_item``
for the updated position.
.. versionchanged:: 8.0
Added the ``current_item`` optional parameter.
.. versionchanged:: 8.0
Only render when the number of steps meets the
``update_min_steps`` threshold.
"""
if current_item is not None:
self.current_item = current_item
self._completed_intervals += n_steps
if self._completed_intervals >= self.update_min_steps:
self.make_step(self._completed_intervals)
self.render_progress()
self._completed_intervals = 0
def finish(self) -> None:
self.eta_known = False
self.current_item = None
self.finished = True
def generator(self) -> t.Iterator[V]:
"""Return a generator which yields the items added to the bar
during construction, and updates the progress bar *after* the
yielded block returns.
"""
# WARNING: the iterator interface for `ProgressBar` relies on
# this and only works because this is a simple generator which
# doesn't create or manage additional state. If this function
# changes, the impact should be evaluated both against
# `iter(bar)` and `next(bar)`. `next()` in particular may call
# `self.generator()` repeatedly, and this must remain safe in
# order for that interface to work.
if not self.entered:
raise RuntimeError("You need to use progress bars in a with block.")
if self.is_hidden:
yield from self.iter
else:
for rv in self.iter:
self.current_item = rv
# This allows show_item_func to be updated before the
# item is processed. Only trigger at the beginning of
# the update interval.
if self._completed_intervals == 0:
self.render_progress()
yield rv
self.update(1)
self.finish()
self.render_progress()
def pager(generator: t.Iterable[str], color: t.Optional[bool] = None) -> None:
"""Decide what method to use for paging through text."""
stdout = _default_text_stdout()
if not isatty(sys.stdin) or not isatty(stdout):
return _nullpager(stdout, generator, color)
pager_cmd = (os.environ.get("PAGER", None) or "").strip()
if pager_cmd:
if WIN:
return _tempfilepager(generator, pager_cmd, color)
return _pipepager(generator, pager_cmd, color)
if os.environ.get("TERM") in ("dumb", "emacs"):
return _nullpager(stdout, generator, color)
if WIN or sys.platform.startswith("os2"):
return _tempfilepager(generator, "more <", color)
if hasattr(os, "system") and os.system("(less) 2>/dev/null") == 0:
return _pipepager(generator, "less", color)
import tempfile
fd, filename = tempfile.mkstemp()
os.close(fd)
try:
if hasattr(os, "system") and os.system(f'more "{filename}"') == 0:
return _pipepager(generator, "more", color)
return _nullpager(stdout, generator, color)
finally:
os.unlink(filename)
def _pipepager(generator: t.Iterable[str], cmd: str, color: t.Optional[bool]) -> None:
"""Page through text by feeding it to another program. Invoking a
pager through this might support colors.
"""
import subprocess
env = dict(os.environ)
# If we're piping to less we might support colors under the
# condition that
cmd_detail = cmd.rsplit("/", 1)[-1].split()
if color is None and cmd_detail[0] == "less":
less_flags = f"{os.environ.get('LESS', '')}{' '.join(cmd_detail[1:])}"
if not less_flags:
env["LESS"] = "-R"
color = True
elif "r" in less_flags or "R" in less_flags:
color = True
c = subprocess.Popen(cmd, shell=True, stdin=subprocess.PIPE, env=env)
stdin = t.cast(t.BinaryIO, c.stdin)
encoding = get_best_encoding(stdin)
try:
for text in generator:
if not color:
text = strip_ansi(text)
stdin.write(text.encode(encoding, "replace"))
except (OSError, KeyboardInterrupt):
pass
else:
stdin.close()
# Less doesn't respect ^C, but catches it for its own UI purposes (aborting
# search or other commands inside less).
#
# That means when the user hits ^C, the parent process (click) terminates,
# but less is still alive, paging the output and messing up the terminal.
#
# If the user wants to make the pager exit on ^C, they should set
# `LESS='-K'`. It's not our decision to make.
while True:
try:
c.wait()
except KeyboardInterrupt:
pass
else:
break
def _tempfilepager(
generator: t.Iterable[str], cmd: str, color: t.Optional[bool]
) -> None:
"""Page through text by invoking a program on a temporary file."""
import tempfile
fd, filename = tempfile.mkstemp()
# TODO: This never terminates if the passed generator never terminates.
text = "".join(generator)
if not color:
text = strip_ansi(text)
encoding = get_best_encoding(sys.stdout)
with open_stream(filename, "wb")[0] as f:
f.write(text.encode(encoding))
try:
os.system(f'{cmd} "{filename}"')
finally:
os.close(fd)
os.unlink(filename)
def _nullpager(
stream: t.TextIO, generator: t.Iterable[str], color: t.Optional[bool]
) -> None:
"""Simply print unformatted text. This is the ultimate fallback."""
for text in generator:
if not color:
text = strip_ansi(text)
stream.write(text)
class Editor:
def __init__(
self,
editor: t.Optional[str] = None,
env: t.Optional[t.Mapping[str, str]] = None,
require_save: bool = True,
extension: str = ".txt",
) -> None:
self.editor = editor
self.env = env
self.require_save = require_save
self.extension = extension
def get_editor(self) -> str:
if self.editor is not None:
return self.editor
for key in "VISUAL", "EDITOR":
rv = os.environ.get(key)
if rv:
return rv
if WIN:
return "notepad"
for editor in "sensible-editor", "vim", "nano":
if os.system(f"which {editor} >/dev/null 2>&1") == 0:
return editor
return "vi"
def edit_file(self, filename: str) -> None:
import subprocess
editor = self.get_editor()
environ: t.Optional[t.Dict[str, str]] = None
if self.env:
environ = os.environ.copy()
environ.update(self.env)
try:
c = subprocess.Popen(f'{editor} "{filename}"', env=environ, shell=True)
exit_code = c.wait()
if exit_code != 0:
raise ClickException(
_("{editor}: Editing failed").format(editor=editor)
)
except OSError as e:
raise ClickException(
_("{editor}: Editing failed: {e}").format(editor=editor, e=e)
) from e
def edit(self, text: t.Optional[t.AnyStr]) -> t.Optional[t.AnyStr]:
import tempfile
if not text:
data = b""
elif isinstance(text, (bytes, bytearray)):
data = text
else:
if text and not text.endswith("\n"):
text += "\n"
if WIN:
data = text.replace("\n", "\r\n").encode("utf-8-sig")
else:
data = text.encode("utf-8")
fd, name = tempfile.mkstemp(prefix="editor-", suffix=self.extension)
f: t.BinaryIO
try:
with os.fdopen(fd, "wb") as f:
f.write(data)
# If the filesystem resolution is 1 second, like Mac OS
# 10.12 Extended, or 2 seconds, like FAT32, and the editor
# closes very fast, require_save can fail. Set the modified
# time to be 2 seconds in the past to work around this.
os.utime(name, (os.path.getatime(name), os.path.getmtime(name) - 2))
# Depending on the resolution, the exact value might not be
# recorded, so get the new recorded value.
timestamp = os.path.getmtime(name)
self.edit_file(name)
if self.require_save and os.path.getmtime(name) == timestamp:
return None
with open(name, "rb") as f:
rv = f.read()
if isinstance(text, (bytes, bytearray)):
return rv
return rv.decode("utf-8-sig").replace("\r\n", "\n") # type: ignore
finally:
os.unlink(name)
def open_url(url: str, wait: bool = False, locate: bool = False) -> int:
import subprocess
def _unquote_file(url: str) -> str:
from urllib.parse import unquote
if url.startswith("file://"):
url = unquote(url[7:])
return url
if sys.platform == "darwin":
args = ["open"]
if wait:
args.append("-W")
if locate:
args.append("-R")
args.append(_unquote_file(url))
null = open("/dev/null", "w")
try:
return subprocess.Popen(args, stderr=null).wait()
finally:
null.close()
elif WIN:
if locate:
url = _unquote_file(url.replace('"', ""))
args = f'explorer /select,"{url}"'
else:
url = url.replace('"', "")
wait_str = "/WAIT" if wait else ""
args = f'start {wait_str} "" "{url}"'
return os.system(args)
elif CYGWIN:
if locate:
url = os.path.dirname(_unquote_file(url).replace('"', ""))
args = f'cygstart "{url}"'
else:
url = url.replace('"', "")
wait_str = "-w" if wait else ""
args = f'cygstart {wait_str} "{url}"'
return os.system(args)
try:
if locate:
url = os.path.dirname(_unquote_file(url)) or "."
else:
url = _unquote_file(url)
c = subprocess.Popen(["xdg-open", url])
if wait:
return c.wait()
return 0
except OSError:
if url.startswith(("http://", "https://")) and not locate and not wait:
import webbrowser
webbrowser.open(url)
return 0
return 1
def _translate_ch_to_exc(ch: str) -> t.Optional[BaseException]:
if ch == "\x03":
raise KeyboardInterrupt()
if ch == "\x04" and not WIN: # Unix-like, Ctrl+D
raise EOFError()
if ch == "\x1a" and WIN: # Windows, Ctrl+Z
raise EOFError()
return None
if WIN:
import msvcrt
@contextlib.contextmanager
def raw_terminal() -> t.Iterator[int]:
yield -1
def getchar(echo: bool) -> str:
# The function `getch` will return a bytes object corresponding to
# the pressed character. Since Windows 10 build 1803, it will also
# return \x00 when called a second time after pressing a regular key.
#
# `getwch` does not share this probably-bugged behavior. Moreover, it
# returns a Unicode object by default, which is what we want.
#
# Either of these functions will return \x00 or \xe0 to indicate
# a special key, and you need to call the same function again to get
# the "rest" of the code. The fun part is that \u00e0 is
# "latin small letter a with grave", so if you type that on a French
# keyboard, you _also_ get a \xe0.
# E.g., consider the Up arrow. This returns \xe0 and then \x48. The
# resulting Unicode string reads as "a with grave" + "capital H".
# This is indistinguishable from when the user actually types
# "a with grave" and then "capital H".
#
# When \xe0 is returned, we assume it's part of a special-key sequence
# and call `getwch` again, but that means that when the user types
# the \u00e0 character, `getchar` doesn't return until a second
# character is typed.
# The alternative is returning immediately, but that would mess up
# cross-platform handling of arrow keys and others that start with
# \xe0. Another option is using `getch`, but then we can't reliably
# read non-ASCII characters, because return values of `getch` are
# limited to the current 8-bit codepage.
#
# Anyway, Click doesn't claim to do this Right(tm), and using `getwch`
# is doing the right thing in more situations than with `getch`.
func: t.Callable[[], str]
if echo:
func = msvcrt.getwche # type: ignore
else:
func = msvcrt.getwch # type: ignore
rv = func()
if rv in ("\x00", "\xe0"):
# \x00 and \xe0 are control characters that indicate special key,
# see above.
rv += func()
_translate_ch_to_exc(rv)
return rv
else:
import tty
import termios
@contextlib.contextmanager
def raw_terminal() -> t.Iterator[int]:
f: t.Optional[t.TextIO]
fd: int
if not isatty(sys.stdin):
f = open("/dev/tty")
fd = f.fileno()
else:
fd = sys.stdin.fileno()
f = None
try:
old_settings = termios.tcgetattr(fd)
try:
tty.setraw(fd)
yield fd
finally:
termios.tcsetattr(fd, termios.TCSADRAIN, old_settings)
sys.stdout.flush()
if f is not None:
f.close()
except termios.error:
pass
def getchar(echo: bool) -> str:
with raw_terminal() as fd:
ch = os.read(fd, 32).decode(get_best_encoding(sys.stdin), "replace")
if echo and isatty(sys.stdout):
sys.stdout.write(ch)
_translate_ch_to_exc(ch)
return ch

View File

@ -0,0 +1,49 @@
import textwrap
import typing as t
from contextlib import contextmanager
class TextWrapper(textwrap.TextWrapper):
def _handle_long_word(
self,
reversed_chunks: t.List[str],
cur_line: t.List[str],
cur_len: int,
width: int,
) -> None:
space_left = max(width - cur_len, 1)
if self.break_long_words:
last = reversed_chunks[-1]
cut = last[:space_left]
res = last[space_left:]
cur_line.append(cut)
reversed_chunks[-1] = res
elif not cur_line:
cur_line.append(reversed_chunks.pop())
@contextmanager
def extra_indent(self, indent: str) -> t.Iterator[None]:
old_initial_indent = self.initial_indent
old_subsequent_indent = self.subsequent_indent
self.initial_indent += indent
self.subsequent_indent += indent
try:
yield
finally:
self.initial_indent = old_initial_indent
self.subsequent_indent = old_subsequent_indent
def indent_only(self, text: str) -> str:
rv = []
for idx, line in enumerate(text.splitlines()):
indent = self.initial_indent
if idx > 0:
indent = self.subsequent_indent
rv.append(f"{indent}{line}")
return "\n".join(rv)

View File

@ -0,0 +1,100 @@
import codecs
import os
from gettext import gettext as _
def _verify_python_env() -> None:
"""Ensures that the environment is good for Unicode."""
try:
from locale import getpreferredencoding
fs_enc = codecs.lookup(getpreferredencoding()).name
except Exception:
fs_enc = "ascii"
if fs_enc != "ascii":
return
extra = [
_(
"Click will abort further execution because Python was"
" configured to use ASCII as encoding for the environment."
" Consult https://click.palletsprojects.com/unicode-support/"
" for mitigation steps."
)
]
if os.name == "posix":
import subprocess
try:
rv = subprocess.Popen(
["locale", "-a"],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
encoding="ascii",
errors="replace",
).communicate()[0]
except OSError:
rv = ""
good_locales = set()
has_c_utf8 = False
for line in rv.splitlines():
locale = line.strip()
if locale.lower().endswith((".utf-8", ".utf8")):
good_locales.add(locale)
if locale.lower() in ("c.utf8", "c.utf-8"):
has_c_utf8 = True
if not good_locales:
extra.append(
_(
"Additional information: on this system no suitable"
" UTF-8 locales were discovered. This most likely"
" requires resolving by reconfiguring the locale"
" system."
)
)
elif has_c_utf8:
extra.append(
_(
"This system supports the C.UTF-8 locale which is"
" recommended. You might be able to resolve your"
" issue by exporting the following environment"
" variables:"
)
)
extra.append(" export LC_ALL=C.UTF-8\n export LANG=C.UTF-8")
else:
extra.append(
_(
"This system lists some UTF-8 supporting locales"
" that you can pick from. The following suitable"
" locales were discovered: {locales}"
).format(locales=", ".join(sorted(good_locales)))
)
bad_locale = None
for env_locale in os.environ.get("LC_ALL"), os.environ.get("LANG"):
if env_locale and env_locale.lower().endswith((".utf-8", ".utf8")):
bad_locale = env_locale
if env_locale is not None:
break
if bad_locale is not None:
extra.append(
_(
"Click discovered that you exported a UTF-8 locale"
" but the locale system could not pick up from it"
" because it does not exist. The exported locale is"
" {locale!r} but it is not supported."
).format(locale=bad_locale)
)
raise RuntimeError("\n\n".join(extra))

View File

@ -0,0 +1,279 @@
# This module is based on the excellent work by Adam Bartoš who
# provided a lot of what went into the implementation here in
# the discussion to issue1602 in the Python bug tracker.
#
# There are some general differences in regards to how this works
# compared to the original patches as we do not need to patch
# the entire interpreter but just work in our little world of
# echo and prompt.
import io
import sys
import time
import typing as t
from ctypes import byref
from ctypes import c_char
from ctypes import c_char_p
from ctypes import c_int
from ctypes import c_ssize_t
from ctypes import c_ulong
from ctypes import c_void_p
from ctypes import POINTER
from ctypes import py_object
from ctypes import Structure
from ctypes.wintypes import DWORD
from ctypes.wintypes import HANDLE
from ctypes.wintypes import LPCWSTR
from ctypes.wintypes import LPWSTR
from ._compat import _NonClosingTextIOWrapper
assert sys.platform == "win32"
import msvcrt # noqa: E402
from ctypes import windll # noqa: E402
from ctypes import WINFUNCTYPE # noqa: E402
c_ssize_p = POINTER(c_ssize_t)
kernel32 = windll.kernel32
GetStdHandle = kernel32.GetStdHandle
ReadConsoleW = kernel32.ReadConsoleW
WriteConsoleW = kernel32.WriteConsoleW
GetConsoleMode = kernel32.GetConsoleMode
GetLastError = kernel32.GetLastError
GetCommandLineW = WINFUNCTYPE(LPWSTR)(("GetCommandLineW", windll.kernel32))
CommandLineToArgvW = WINFUNCTYPE(POINTER(LPWSTR), LPCWSTR, POINTER(c_int))(
("CommandLineToArgvW", windll.shell32)
)
LocalFree = WINFUNCTYPE(c_void_p, c_void_p)(("LocalFree", windll.kernel32))
STDIN_HANDLE = GetStdHandle(-10)
STDOUT_HANDLE = GetStdHandle(-11)
STDERR_HANDLE = GetStdHandle(-12)
PyBUF_SIMPLE = 0
PyBUF_WRITABLE = 1
ERROR_SUCCESS = 0
ERROR_NOT_ENOUGH_MEMORY = 8
ERROR_OPERATION_ABORTED = 995
STDIN_FILENO = 0
STDOUT_FILENO = 1
STDERR_FILENO = 2
EOF = b"\x1a"
MAX_BYTES_WRITTEN = 32767
try:
from ctypes import pythonapi
except ImportError:
# On PyPy we cannot get buffers so our ability to operate here is
# severely limited.
get_buffer = None
else:
class Py_buffer(Structure):
_fields_ = [
("buf", c_void_p),
("obj", py_object),
("len", c_ssize_t),
("itemsize", c_ssize_t),
("readonly", c_int),
("ndim", c_int),
("format", c_char_p),
("shape", c_ssize_p),
("strides", c_ssize_p),
("suboffsets", c_ssize_p),
("internal", c_void_p),
]
PyObject_GetBuffer = pythonapi.PyObject_GetBuffer
PyBuffer_Release = pythonapi.PyBuffer_Release
def get_buffer(obj, writable=False):
buf = Py_buffer()
flags = PyBUF_WRITABLE if writable else PyBUF_SIMPLE
PyObject_GetBuffer(py_object(obj), byref(buf), flags)
try:
buffer_type = c_char * buf.len
return buffer_type.from_address(buf.buf)
finally:
PyBuffer_Release(byref(buf))
class _WindowsConsoleRawIOBase(io.RawIOBase):
def __init__(self, handle):
self.handle = handle
def isatty(self):
super().isatty()
return True
class _WindowsConsoleReader(_WindowsConsoleRawIOBase):
def readable(self):
return True
def readinto(self, b):
bytes_to_be_read = len(b)
if not bytes_to_be_read:
return 0
elif bytes_to_be_read % 2:
raise ValueError(
"cannot read odd number of bytes from UTF-16-LE encoded console"
)
buffer = get_buffer(b, writable=True)
code_units_to_be_read = bytes_to_be_read // 2
code_units_read = c_ulong()
rv = ReadConsoleW(
HANDLE(self.handle),
buffer,
code_units_to_be_read,
byref(code_units_read),
None,
)
if GetLastError() == ERROR_OPERATION_ABORTED:
# wait for KeyboardInterrupt
time.sleep(0.1)
if not rv:
raise OSError(f"Windows error: {GetLastError()}")
if buffer[0] == EOF:
return 0
return 2 * code_units_read.value
class _WindowsConsoleWriter(_WindowsConsoleRawIOBase):
def writable(self):
return True
@staticmethod
def _get_error_message(errno):
if errno == ERROR_SUCCESS:
return "ERROR_SUCCESS"
elif errno == ERROR_NOT_ENOUGH_MEMORY:
return "ERROR_NOT_ENOUGH_MEMORY"
return f"Windows error {errno}"
def write(self, b):
bytes_to_be_written = len(b)
buf = get_buffer(b)
code_units_to_be_written = min(bytes_to_be_written, MAX_BYTES_WRITTEN) // 2
code_units_written = c_ulong()
WriteConsoleW(
HANDLE(self.handle),
buf,
code_units_to_be_written,
byref(code_units_written),
None,
)
bytes_written = 2 * code_units_written.value
if bytes_written == 0 and bytes_to_be_written > 0:
raise OSError(self._get_error_message(GetLastError()))
return bytes_written
class ConsoleStream:
def __init__(self, text_stream: t.TextIO, byte_stream: t.BinaryIO) -> None:
self._text_stream = text_stream
self.buffer = byte_stream
@property
def name(self) -> str:
return self.buffer.name
def write(self, x: t.AnyStr) -> int:
if isinstance(x, str):
return self._text_stream.write(x)
try:
self.flush()
except Exception:
pass
return self.buffer.write(x)
def writelines(self, lines: t.Iterable[t.AnyStr]) -> None:
for line in lines:
self.write(line)
def __getattr__(self, name: str) -> t.Any:
return getattr(self._text_stream, name)
def isatty(self) -> bool:
return self.buffer.isatty()
def __repr__(self):
return f"<ConsoleStream name={self.name!r} encoding={self.encoding!r}>"
def _get_text_stdin(buffer_stream: t.BinaryIO) -> t.TextIO:
text_stream = _NonClosingTextIOWrapper(
io.BufferedReader(_WindowsConsoleReader(STDIN_HANDLE)),
"utf-16-le",
"strict",
line_buffering=True,
)
return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream))
def _get_text_stdout(buffer_stream: t.BinaryIO) -> t.TextIO:
text_stream = _NonClosingTextIOWrapper(
io.BufferedWriter(_WindowsConsoleWriter(STDOUT_HANDLE)),
"utf-16-le",
"strict",
line_buffering=True,
)
return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream))
def _get_text_stderr(buffer_stream: t.BinaryIO) -> t.TextIO:
text_stream = _NonClosingTextIOWrapper(
io.BufferedWriter(_WindowsConsoleWriter(STDERR_HANDLE)),
"utf-16-le",
"strict",
line_buffering=True,
)
return t.cast(t.TextIO, ConsoleStream(text_stream, buffer_stream))
_stream_factories: t.Mapping[int, t.Callable[[t.BinaryIO], t.TextIO]] = {
0: _get_text_stdin,
1: _get_text_stdout,
2: _get_text_stderr,
}
def _is_console(f: t.TextIO) -> bool:
if not hasattr(f, "fileno"):
return False
try:
fileno = f.fileno()
except (OSError, io.UnsupportedOperation):
return False
handle = msvcrt.get_osfhandle(fileno)
return bool(GetConsoleMode(handle, byref(DWORD())))
def _get_windows_console_stream(
f: t.TextIO, encoding: t.Optional[str], errors: t.Optional[str]
) -> t.Optional[t.TextIO]:
if (
get_buffer is not None
and encoding in {"utf-16-le", None}
and errors in {"strict", None}
and _is_console(f)
):
func = _stream_factories.get(f.fileno())
if func is not None:
b = getattr(f, "buffer", None)
if b is None:
return None
return func(b)

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,436 @@
import inspect
import types
import typing as t
from functools import update_wrapper
from gettext import gettext as _
from .core import Argument
from .core import Command
from .core import Context
from .core import Group
from .core import Option
from .core import Parameter
from .globals import get_current_context
from .utils import echo
F = t.TypeVar("F", bound=t.Callable[..., t.Any])
FC = t.TypeVar("FC", t.Callable[..., t.Any], Command)
def pass_context(f: F) -> F:
"""Marks a callback as wanting to receive the current context
object as first argument.
"""
def new_func(*args, **kwargs): # type: ignore
return f(get_current_context(), *args, **kwargs)
return update_wrapper(t.cast(F, new_func), f)
def pass_obj(f: F) -> F:
"""Similar to :func:`pass_context`, but only pass the object on the
context onwards (:attr:`Context.obj`). This is useful if that object
represents the state of a nested system.
"""
def new_func(*args, **kwargs): # type: ignore
return f(get_current_context().obj, *args, **kwargs)
return update_wrapper(t.cast(F, new_func), f)
def make_pass_decorator(
object_type: t.Type, ensure: bool = False
) -> "t.Callable[[F], F]":
"""Given an object type this creates a decorator that will work
similar to :func:`pass_obj` but instead of passing the object of the
current context, it will find the innermost context of type
:func:`object_type`.
This generates a decorator that works roughly like this::
from functools import update_wrapper
def decorator(f):
@pass_context
def new_func(ctx, *args, **kwargs):
obj = ctx.find_object(object_type)
return ctx.invoke(f, obj, *args, **kwargs)
return update_wrapper(new_func, f)
return decorator
:param object_type: the type of the object to pass.
:param ensure: if set to `True`, a new object will be created and
remembered on the context if it's not there yet.
"""
def decorator(f: F) -> F:
def new_func(*args, **kwargs): # type: ignore
ctx = get_current_context()
if ensure:
obj = ctx.ensure_object(object_type)
else:
obj = ctx.find_object(object_type)
if obj is None:
raise RuntimeError(
"Managed to invoke callback without a context"
f" object of type {object_type.__name__!r}"
" existing."
)
return ctx.invoke(f, obj, *args, **kwargs)
return update_wrapper(t.cast(F, new_func), f)
return decorator
def pass_meta_key(
key: str, *, doc_description: t.Optional[str] = None
) -> "t.Callable[[F], F]":
"""Create a decorator that passes a key from
:attr:`click.Context.meta` as the first argument to the decorated
function.
:param key: Key in ``Context.meta`` to pass.
:param doc_description: Description of the object being passed,
inserted into the decorator's docstring. Defaults to "the 'key'
key from Context.meta".
.. versionadded:: 8.0
"""
def decorator(f: F) -> F:
def new_func(*args, **kwargs): # type: ignore
ctx = get_current_context()
obj = ctx.meta[key]
return ctx.invoke(f, obj, *args, **kwargs)
return update_wrapper(t.cast(F, new_func), f)
if doc_description is None:
doc_description = f"the {key!r} key from :attr:`click.Context.meta`"
decorator.__doc__ = (
f"Decorator that passes {doc_description} as the first argument"
" to the decorated function."
)
return decorator
def _make_command(
f: F,
name: t.Optional[str],
attrs: t.MutableMapping[str, t.Any],
cls: t.Type[Command],
) -> Command:
if isinstance(f, Command):
raise TypeError("Attempted to convert a callback into a command twice.")
try:
params = f.__click_params__ # type: ignore
params.reverse()
del f.__click_params__ # type: ignore
except AttributeError:
params = []
help = attrs.get("help")
if help is None:
help = inspect.getdoc(f)
else:
help = inspect.cleandoc(help)
attrs["help"] = help
return cls(
name=name or f.__name__.lower().replace("_", "-"),
callback=f,
params=params,
**attrs,
)
def command(
name: t.Optional[str] = None,
cls: t.Optional[t.Type[Command]] = None,
**attrs: t.Any,
) -> t.Callable[[F], Command]:
r"""Creates a new :class:`Command` and uses the decorated function as
callback. This will also automatically attach all decorated
:func:`option`\s and :func:`argument`\s as parameters to the command.
The name of the command defaults to the name of the function with
underscores replaced by dashes. If you want to change that, you can
pass the intended name as the first argument.
All keyword arguments are forwarded to the underlying command class.
Once decorated the function turns into a :class:`Command` instance
that can be invoked as a command line utility or be attached to a
command :class:`Group`.
:param name: the name of the command. This defaults to the function
name with underscores replaced by dashes.
:param cls: the command class to instantiate. This defaults to
:class:`Command`.
"""
if cls is None:
cls = Command
def decorator(f: t.Callable[..., t.Any]) -> Command:
cmd = _make_command(f, name, attrs, cls) # type: ignore
cmd.__doc__ = f.__doc__
return cmd
return decorator
def group(name: t.Optional[str] = None, **attrs: t.Any) -> t.Callable[[F], Group]:
"""Creates a new :class:`Group` with a function as callback. This
works otherwise the same as :func:`command` just that the `cls`
parameter is set to :class:`Group`.
"""
attrs.setdefault("cls", Group)
return t.cast(Group, command(name, **attrs))
def _param_memo(f: FC, param: Parameter) -> None:
if isinstance(f, Command):
f.params.append(param)
else:
if not hasattr(f, "__click_params__"):
f.__click_params__ = [] # type: ignore
f.__click_params__.append(param) # type: ignore
def argument(*param_decls: str, **attrs: t.Any) -> t.Callable[[FC], FC]:
"""Attaches an argument to the command. All positional arguments are
passed as parameter declarations to :class:`Argument`; all keyword
arguments are forwarded unchanged (except ``cls``).
This is equivalent to creating an :class:`Argument` instance manually
and attaching it to the :attr:`Command.params` list.
:param cls: the argument class to instantiate. This defaults to
:class:`Argument`.
"""
def decorator(f: FC) -> FC:
ArgumentClass = attrs.pop("cls", Argument)
_param_memo(f, ArgumentClass(param_decls, **attrs))
return f
return decorator
def option(*param_decls: str, **attrs: t.Any) -> t.Callable[[FC], FC]:
"""Attaches an option to the command. All positional arguments are
passed as parameter declarations to :class:`Option`; all keyword
arguments are forwarded unchanged (except ``cls``).
This is equivalent to creating an :class:`Option` instance manually
and attaching it to the :attr:`Command.params` list.
:param cls: the option class to instantiate. This defaults to
:class:`Option`.
"""
def decorator(f: FC) -> FC:
# Issue 926, copy attrs, so pre-defined options can re-use the same cls=
option_attrs = attrs.copy()
if "help" in option_attrs:
option_attrs["help"] = inspect.cleandoc(option_attrs["help"])
OptionClass = option_attrs.pop("cls", Option)
_param_memo(f, OptionClass(param_decls, **option_attrs))
return f
return decorator
def confirmation_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]:
"""Add a ``--yes`` option which shows a prompt before continuing if
not passed. If the prompt is declined, the program will exit.
:param param_decls: One or more option names. Defaults to the single
value ``"--yes"``.
:param kwargs: Extra arguments are passed to :func:`option`.
"""
def callback(ctx: Context, param: Parameter, value: bool) -> None:
if not value:
ctx.abort()
if not param_decls:
param_decls = ("--yes",)
kwargs.setdefault("is_flag", True)
kwargs.setdefault("callback", callback)
kwargs.setdefault("expose_value", False)
kwargs.setdefault("prompt", "Do you want to continue?")
kwargs.setdefault("help", "Confirm the action without prompting.")
return option(*param_decls, **kwargs)
def password_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]:
"""Add a ``--password`` option which prompts for a password, hiding
input and asking to enter the value again for confirmation.
:param param_decls: One or more option names. Defaults to the single
value ``"--password"``.
:param kwargs: Extra arguments are passed to :func:`option`.
"""
if not param_decls:
param_decls = ("--password",)
kwargs.setdefault("prompt", True)
kwargs.setdefault("confirmation_prompt", True)
kwargs.setdefault("hide_input", True)
return option(*param_decls, **kwargs)
def version_option(
version: t.Optional[str] = None,
*param_decls: str,
package_name: t.Optional[str] = None,
prog_name: t.Optional[str] = None,
message: t.Optional[str] = None,
**kwargs: t.Any,
) -> t.Callable[[FC], FC]:
"""Add a ``--version`` option which immediately prints the version
number and exits the program.
If ``version`` is not provided, Click will try to detect it using
:func:`importlib.metadata.version` to get the version for the
``package_name``. On Python < 3.8, the ``importlib_metadata``
backport must be installed.
If ``package_name`` is not provided, Click will try to detect it by
inspecting the stack frames. This will be used to detect the
version, so it must match the name of the installed package.
:param version: The version number to show. If not provided, Click
will try to detect it.
:param param_decls: One or more option names. Defaults to the single
value ``"--version"``.
:param package_name: The package name to detect the version from. If
not provided, Click will try to detect it.
:param prog_name: The name of the CLI to show in the message. If not
provided, it will be detected from the command.
:param message: The message to show. The values ``%(prog)s``,
``%(package)s``, and ``%(version)s`` are available. Defaults to
``"%(prog)s, version %(version)s"``.
:param kwargs: Extra arguments are passed to :func:`option`.
:raise RuntimeError: ``version`` could not be detected.
.. versionchanged:: 8.0
Add the ``package_name`` parameter, and the ``%(package)s``
value for messages.
.. versionchanged:: 8.0
Use :mod:`importlib.metadata` instead of ``pkg_resources``. The
version is detected based on the package name, not the entry
point name. The Python package name must match the installed
package name, or be passed with ``package_name=``.
"""
if message is None:
message = _("%(prog)s, version %(version)s")
if version is None and package_name is None:
frame = inspect.currentframe()
f_back = frame.f_back if frame is not None else None
f_globals = f_back.f_globals if f_back is not None else None
# break reference cycle
# https://docs.python.org/3/library/inspect.html#the-interpreter-stack
del frame
if f_globals is not None:
package_name = f_globals.get("__name__")
if package_name == "__main__":
package_name = f_globals.get("__package__")
if package_name:
package_name = package_name.partition(".")[0]
def callback(ctx: Context, param: Parameter, value: bool) -> None:
if not value or ctx.resilient_parsing:
return
nonlocal prog_name
nonlocal version
if prog_name is None:
prog_name = ctx.find_root().info_name
if version is None and package_name is not None:
metadata: t.Optional[types.ModuleType]
try:
from importlib import metadata # type: ignore
except ImportError:
# Python < 3.8
import importlib_metadata as metadata # type: ignore
try:
version = metadata.version(package_name) # type: ignore
except metadata.PackageNotFoundError: # type: ignore
raise RuntimeError(
f"{package_name!r} is not installed. Try passing"
" 'package_name' instead."
) from None
if version is None:
raise RuntimeError(
f"Could not determine the version for {package_name!r} automatically."
)
echo(
t.cast(str, message)
% {"prog": prog_name, "package": package_name, "version": version},
color=ctx.color,
)
ctx.exit()
if not param_decls:
param_decls = ("--version",)
kwargs.setdefault("is_flag", True)
kwargs.setdefault("expose_value", False)
kwargs.setdefault("is_eager", True)
kwargs.setdefault("help", _("Show the version and exit."))
kwargs["callback"] = callback
return option(*param_decls, **kwargs)
def help_option(*param_decls: str, **kwargs: t.Any) -> t.Callable[[FC], FC]:
"""Add a ``--help`` option which immediately prints the help page
and exits the program.
This is usually unnecessary, as the ``--help`` option is added to
each command automatically unless ``add_help_option=False`` is
passed.
:param param_decls: One or more option names. Defaults to the single
value ``"--help"``.
:param kwargs: Extra arguments are passed to :func:`option`.
"""
def callback(ctx: Context, param: Parameter, value: bool) -> None:
if not value or ctx.resilient_parsing:
return
echo(ctx.get_help(), color=ctx.color)
ctx.exit()
if not param_decls:
param_decls = ("--help",)
kwargs.setdefault("is_flag", True)
kwargs.setdefault("expose_value", False)
kwargs.setdefault("is_eager", True)
kwargs.setdefault("help", _("Show this message and exit."))
kwargs["callback"] = callback
return option(*param_decls, **kwargs)

View File

@ -0,0 +1,287 @@
import os
import typing as t
from gettext import gettext as _
from gettext import ngettext
from ._compat import get_text_stderr
from .utils import echo
if t.TYPE_CHECKING:
from .core import Context
from .core import Parameter
def _join_param_hints(
param_hint: t.Optional[t.Union[t.Sequence[str], str]]
) -> t.Optional[str]:
if param_hint is not None and not isinstance(param_hint, str):
return " / ".join(repr(x) for x in param_hint)
return param_hint
class ClickException(Exception):
"""An exception that Click can handle and show to the user."""
#: The exit code for this exception.
exit_code = 1
def __init__(self, message: str) -> None:
super().__init__(message)
self.message = message
def format_message(self) -> str:
return self.message
def __str__(self) -> str:
return self.message
def show(self, file: t.Optional[t.IO] = None) -> None:
if file is None:
file = get_text_stderr()
echo(_("Error: {message}").format(message=self.format_message()), file=file)
class UsageError(ClickException):
"""An internal exception that signals a usage error. This typically
aborts any further handling.
:param message: the error message to display.
:param ctx: optionally the context that caused this error. Click will
fill in the context automatically in some situations.
"""
exit_code = 2
def __init__(self, message: str, ctx: t.Optional["Context"] = None) -> None:
super().__init__(message)
self.ctx = ctx
self.cmd = self.ctx.command if self.ctx else None
def show(self, file: t.Optional[t.IO] = None) -> None:
if file is None:
file = get_text_stderr()
color = None
hint = ""
if (
self.ctx is not None
and self.ctx.command.get_help_option(self.ctx) is not None
):
hint = _("Try '{command} {option}' for help.").format(
command=self.ctx.command_path, option=self.ctx.help_option_names[0]
)
hint = f"{hint}\n"
if self.ctx is not None:
color = self.ctx.color
echo(f"{self.ctx.get_usage()}\n{hint}", file=file, color=color)
echo(
_("Error: {message}").format(message=self.format_message()),
file=file,
color=color,
)
class BadParameter(UsageError):
"""An exception that formats out a standardized error message for a
bad parameter. This is useful when thrown from a callback or type as
Click will attach contextual information to it (for instance, which
parameter it is).
.. versionadded:: 2.0
:param param: the parameter object that caused this error. This can
be left out, and Click will attach this info itself
if possible.
:param param_hint: a string that shows up as parameter name. This
can be used as alternative to `param` in cases
where custom validation should happen. If it is
a string it's used as such, if it's a list then
each item is quoted and separated.
"""
def __init__(
self,
message: str,
ctx: t.Optional["Context"] = None,
param: t.Optional["Parameter"] = None,
param_hint: t.Optional[str] = None,
) -> None:
super().__init__(message, ctx)
self.param = param
self.param_hint = param_hint
def format_message(self) -> str:
if self.param_hint is not None:
param_hint = self.param_hint
elif self.param is not None:
param_hint = self.param.get_error_hint(self.ctx) # type: ignore
else:
return _("Invalid value: {message}").format(message=self.message)
return _("Invalid value for {param_hint}: {message}").format(
param_hint=_join_param_hints(param_hint), message=self.message
)
class MissingParameter(BadParameter):
"""Raised if click required an option or argument but it was not
provided when invoking the script.
.. versionadded:: 4.0
:param param_type: a string that indicates the type of the parameter.
The default is to inherit the parameter type from
the given `param`. Valid values are ``'parameter'``,
``'option'`` or ``'argument'``.
"""
def __init__(
self,
message: t.Optional[str] = None,
ctx: t.Optional["Context"] = None,
param: t.Optional["Parameter"] = None,
param_hint: t.Optional[str] = None,
param_type: t.Optional[str] = None,
) -> None:
super().__init__(message or "", ctx, param, param_hint)
self.param_type = param_type
def format_message(self) -> str:
if self.param_hint is not None:
param_hint: t.Optional[str] = self.param_hint
elif self.param is not None:
param_hint = self.param.get_error_hint(self.ctx) # type: ignore
else:
param_hint = None
param_hint = _join_param_hints(param_hint)
param_hint = f" {param_hint}" if param_hint else ""
param_type = self.param_type
if param_type is None and self.param is not None:
param_type = self.param.param_type_name
msg = self.message
if self.param is not None:
msg_extra = self.param.type.get_missing_message(self.param)
if msg_extra:
if msg:
msg += f". {msg_extra}"
else:
msg = msg_extra
msg = f" {msg}" if msg else ""
# Translate param_type for known types.
if param_type == "argument":
missing = _("Missing argument")
elif param_type == "option":
missing = _("Missing option")
elif param_type == "parameter":
missing = _("Missing parameter")
else:
missing = _("Missing {param_type}").format(param_type=param_type)
return f"{missing}{param_hint}.{msg}"
def __str__(self) -> str:
if not self.message:
param_name = self.param.name if self.param else None
return _("Missing parameter: {param_name}").format(param_name=param_name)
else:
return self.message
class NoSuchOption(UsageError):
"""Raised if click attempted to handle an option that does not
exist.
.. versionadded:: 4.0
"""
def __init__(
self,
option_name: str,
message: t.Optional[str] = None,
possibilities: t.Optional[t.Sequence[str]] = None,
ctx: t.Optional["Context"] = None,
) -> None:
if message is None:
message = _("No such option: {name}").format(name=option_name)
super().__init__(message, ctx)
self.option_name = option_name
self.possibilities = possibilities
def format_message(self) -> str:
if not self.possibilities:
return self.message
possibility_str = ", ".join(sorted(self.possibilities))
suggest = ngettext(
"Did you mean {possibility}?",
"(Possible options: {possibilities})",
len(self.possibilities),
).format(possibility=possibility_str, possibilities=possibility_str)
return f"{self.message} {suggest}"
class BadOptionUsage(UsageError):
"""Raised if an option is generally supplied but the use of the option
was incorrect. This is for instance raised if the number of arguments
for an option is not correct.
.. versionadded:: 4.0
:param option_name: the name of the option being used incorrectly.
"""
def __init__(
self, option_name: str, message: str, ctx: t.Optional["Context"] = None
) -> None:
super().__init__(message, ctx)
self.option_name = option_name
class BadArgumentUsage(UsageError):
"""Raised if an argument is generally supplied but the use of the argument
was incorrect. This is for instance raised if the number of values
for an argument is not correct.
.. versionadded:: 6.0
"""
class FileError(ClickException):
"""Raised if a file cannot be opened."""
def __init__(self, filename: str, hint: t.Optional[str] = None) -> None:
if hint is None:
hint = _("unknown error")
super().__init__(hint)
self.ui_filename = os.fsdecode(filename)
self.filename = filename
def format_message(self) -> str:
return _("Could not open file {filename!r}: {message}").format(
filename=self.ui_filename, message=self.message
)
class Abort(RuntimeError):
"""An internal signalling exception that signals Click to abort."""
class Exit(RuntimeError):
"""An exception that indicates that the application should exit with some
status code.
:param code: the status code to exit with.
"""
__slots__ = ("exit_code",)
def __init__(self, code: int = 0) -> None:
self.exit_code = code

View File

@ -0,0 +1,301 @@
import typing as t
from contextlib import contextmanager
from gettext import gettext as _
from ._compat import term_len
from .parser import split_opt
# Can force a width. This is used by the test system
FORCED_WIDTH: t.Optional[int] = None
def measure_table(rows: t.Iterable[t.Tuple[str, str]]) -> t.Tuple[int, ...]:
widths: t.Dict[int, int] = {}
for row in rows:
for idx, col in enumerate(row):
widths[idx] = max(widths.get(idx, 0), term_len(col))
return tuple(y for x, y in sorted(widths.items()))
def iter_rows(
rows: t.Iterable[t.Tuple[str, str]], col_count: int
) -> t.Iterator[t.Tuple[str, ...]]:
for row in rows:
yield row + ("",) * (col_count - len(row))
def wrap_text(
text: str,
width: int = 78,
initial_indent: str = "",
subsequent_indent: str = "",
preserve_paragraphs: bool = False,
) -> str:
"""A helper function that intelligently wraps text. By default, it
assumes that it operates on a single paragraph of text but if the
`preserve_paragraphs` parameter is provided it will intelligently
handle paragraphs (defined by two empty lines).
If paragraphs are handled, a paragraph can be prefixed with an empty
line containing the ``\\b`` character (``\\x08``) to indicate that
no rewrapping should happen in that block.
:param text: the text that should be rewrapped.
:param width: the maximum width for the text.
:param initial_indent: the initial indent that should be placed on the
first line as a string.
:param subsequent_indent: the indent string that should be placed on
each consecutive line.
:param preserve_paragraphs: if this flag is set then the wrapping will
intelligently handle paragraphs.
"""
from ._textwrap import TextWrapper
text = text.expandtabs()
wrapper = TextWrapper(
width,
initial_indent=initial_indent,
subsequent_indent=subsequent_indent,
replace_whitespace=False,
)
if not preserve_paragraphs:
return wrapper.fill(text)
p: t.List[t.Tuple[int, bool, str]] = []
buf: t.List[str] = []
indent = None
def _flush_par() -> None:
if not buf:
return
if buf[0].strip() == "\b":
p.append((indent or 0, True, "\n".join(buf[1:])))
else:
p.append((indent or 0, False, " ".join(buf)))
del buf[:]
for line in text.splitlines():
if not line:
_flush_par()
indent = None
else:
if indent is None:
orig_len = term_len(line)
line = line.lstrip()
indent = orig_len - term_len(line)
buf.append(line)
_flush_par()
rv = []
for indent, raw, text in p:
with wrapper.extra_indent(" " * indent):
if raw:
rv.append(wrapper.indent_only(text))
else:
rv.append(wrapper.fill(text))
return "\n\n".join(rv)
class HelpFormatter:
"""This class helps with formatting text-based help pages. It's
usually just needed for very special internal cases, but it's also
exposed so that developers can write their own fancy outputs.
At present, it always writes into memory.
:param indent_increment: the additional increment for each level.
:param width: the width for the text. This defaults to the terminal
width clamped to a maximum of 78.
"""
def __init__(
self,
indent_increment: int = 2,
width: t.Optional[int] = None,
max_width: t.Optional[int] = None,
) -> None:
import shutil
self.indent_increment = indent_increment
if max_width is None:
max_width = 80
if width is None:
width = FORCED_WIDTH
if width is None:
width = max(min(shutil.get_terminal_size().columns, max_width) - 2, 50)
self.width = width
self.current_indent = 0
self.buffer: t.List[str] = []
def write(self, string: str) -> None:
"""Writes a unicode string into the internal buffer."""
self.buffer.append(string)
def indent(self) -> None:
"""Increases the indentation."""
self.current_indent += self.indent_increment
def dedent(self) -> None:
"""Decreases the indentation."""
self.current_indent -= self.indent_increment
def write_usage(
self, prog: str, args: str = "", prefix: t.Optional[str] = None
) -> None:
"""Writes a usage line into the buffer.
:param prog: the program name.
:param args: whitespace separated list of arguments.
:param prefix: The prefix for the first line. Defaults to
``"Usage: "``.
"""
if prefix is None:
prefix = f"{_('Usage:')} "
usage_prefix = f"{prefix:>{self.current_indent}}{prog} "
text_width = self.width - self.current_indent
if text_width >= (term_len(usage_prefix) + 20):
# The arguments will fit to the right of the prefix.
indent = " " * term_len(usage_prefix)
self.write(
wrap_text(
args,
text_width,
initial_indent=usage_prefix,
subsequent_indent=indent,
)
)
else:
# The prefix is too long, put the arguments on the next line.
self.write(usage_prefix)
self.write("\n")
indent = " " * (max(self.current_indent, term_len(prefix)) + 4)
self.write(
wrap_text(
args, text_width, initial_indent=indent, subsequent_indent=indent
)
)
self.write("\n")
def write_heading(self, heading: str) -> None:
"""Writes a heading into the buffer."""
self.write(f"{'':>{self.current_indent}}{heading}:\n")
def write_paragraph(self) -> None:
"""Writes a paragraph into the buffer."""
if self.buffer:
self.write("\n")
def write_text(self, text: str) -> None:
"""Writes re-indented text into the buffer. This rewraps and
preserves paragraphs.
"""
indent = " " * self.current_indent
self.write(
wrap_text(
text,
self.width,
initial_indent=indent,
subsequent_indent=indent,
preserve_paragraphs=True,
)
)
self.write("\n")
def write_dl(
self,
rows: t.Sequence[t.Tuple[str, str]],
col_max: int = 30,
col_spacing: int = 2,
) -> None:
"""Writes a definition list into the buffer. This is how options
and commands are usually formatted.
:param rows: a list of two item tuples for the terms and values.
:param col_max: the maximum width of the first column.
:param col_spacing: the number of spaces between the first and
second column.
"""
rows = list(rows)
widths = measure_table(rows)
if len(widths) != 2:
raise TypeError("Expected two columns for definition list")
first_col = min(widths[0], col_max) + col_spacing
for first, second in iter_rows(rows, len(widths)):
self.write(f"{'':>{self.current_indent}}{first}")
if not second:
self.write("\n")
continue
if term_len(first) <= first_col - col_spacing:
self.write(" " * (first_col - term_len(first)))
else:
self.write("\n")
self.write(" " * (first_col + self.current_indent))
text_width = max(self.width - first_col - 2, 10)
wrapped_text = wrap_text(second, text_width, preserve_paragraphs=True)
lines = wrapped_text.splitlines()
if lines:
self.write(f"{lines[0]}\n")
for line in lines[1:]:
self.write(f"{'':>{first_col + self.current_indent}}{line}\n")
else:
self.write("\n")
@contextmanager
def section(self, name: str) -> t.Iterator[None]:
"""Helpful context manager that writes a paragraph, a heading,
and the indents.
:param name: the section name that is written as heading.
"""
self.write_paragraph()
self.write_heading(name)
self.indent()
try:
yield
finally:
self.dedent()
@contextmanager
def indentation(self) -> t.Iterator[None]:
"""A context manager that increases the indentation."""
self.indent()
try:
yield
finally:
self.dedent()
def getvalue(self) -> str:
"""Returns the buffer contents."""
return "".join(self.buffer)
def join_options(options: t.Sequence[str]) -> t.Tuple[str, bool]:
"""Given a list of option strings this joins them in the most appropriate
way and returns them in the form ``(formatted_string,
any_prefix_is_slash)`` where the second item in the tuple is a flag that
indicates if any of the option prefixes was a slash.
"""
rv = []
any_prefix_is_slash = False
for opt in options:
prefix = split_opt(opt)[0]
if prefix == "/":
any_prefix_is_slash = True
rv.append((len(prefix), opt))
rv.sort(key=lambda x: x[0])
return ", ".join(x[1] for x in rv), any_prefix_is_slash

View File

@ -0,0 +1,69 @@
import typing
import typing as t
from threading import local
if t.TYPE_CHECKING:
import typing_extensions as te
from .core import Context
_local = local()
@typing.overload
def get_current_context(silent: "te.Literal[False]" = False) -> "Context":
...
@typing.overload
def get_current_context(silent: bool = ...) -> t.Optional["Context"]:
...
def get_current_context(silent: bool = False) -> t.Optional["Context"]:
"""Returns the current click context. This can be used as a way to
access the current context object from anywhere. This is a more implicit
alternative to the :func:`pass_context` decorator. This function is
primarily useful for helpers such as :func:`echo` which might be
interested in changing its behavior based on the current context.
To push the current context, :meth:`Context.scope` can be used.
.. versionadded:: 5.0
:param silent: if set to `True` the return value is `None` if no context
is available. The default behavior is to raise a
:exc:`RuntimeError`.
"""
try:
return t.cast("Context", _local.stack[-1])
except (AttributeError, IndexError) as e:
if not silent:
raise RuntimeError("There is no active click context.") from e
return None
def push_context(ctx: "Context") -> None:
"""Pushes a new context to the current stack."""
_local.__dict__.setdefault("stack", []).append(ctx)
def pop_context() -> None:
"""Removes the top level from the stack."""
_local.stack.pop()
def resolve_color_default(color: t.Optional[bool] = None) -> t.Optional[bool]:
"""Internal helper to get the default value of the color flag. If a
value is passed it's returned unchanged, otherwise it's looked up from
the current context.
"""
if color is not None:
return color
ctx = get_current_context(silent=True)
if ctx is not None:
return ctx.color
return None

View File

@ -0,0 +1,529 @@
"""
This module started out as largely a copy paste from the stdlib's
optparse module with the features removed that we do not need from
optparse because we implement them in Click on a higher level (for
instance type handling, help formatting and a lot more).
The plan is to remove more and more from here over time.
The reason this is a different module and not optparse from the stdlib
is that there are differences in 2.x and 3.x about the error messages
generated and optparse in the stdlib uses gettext for no good reason
and might cause us issues.
Click uses parts of optparse written by Gregory P. Ward and maintained
by the Python Software Foundation. This is limited to code in parser.py.
Copyright 2001-2006 Gregory P. Ward. All rights reserved.
Copyright 2002-2006 Python Software Foundation. All rights reserved.
"""
# This code uses parts of optparse written by Gregory P. Ward and
# maintained by the Python Software Foundation.
# Copyright 2001-2006 Gregory P. Ward
# Copyright 2002-2006 Python Software Foundation
import typing as t
from collections import deque
from gettext import gettext as _
from gettext import ngettext
from .exceptions import BadArgumentUsage
from .exceptions import BadOptionUsage
from .exceptions import NoSuchOption
from .exceptions import UsageError
if t.TYPE_CHECKING:
import typing_extensions as te
from .core import Argument as CoreArgument
from .core import Context
from .core import Option as CoreOption
from .core import Parameter as CoreParameter
V = t.TypeVar("V")
# Sentinel value that indicates an option was passed as a flag without a
# value but is not a flag option. Option.consume_value uses this to
# prompt or use the flag_value.
_flag_needs_value = object()
def _unpack_args(
args: t.Sequence[str], nargs_spec: t.Sequence[int]
) -> t.Tuple[t.Sequence[t.Union[str, t.Sequence[t.Optional[str]], None]], t.List[str]]:
"""Given an iterable of arguments and an iterable of nargs specifications,
it returns a tuple with all the unpacked arguments at the first index
and all remaining arguments as the second.
The nargs specification is the number of arguments that should be consumed
or `-1` to indicate that this position should eat up all the remainders.
Missing items are filled with `None`.
"""
args = deque(args)
nargs_spec = deque(nargs_spec)
rv: t.List[t.Union[str, t.Tuple[t.Optional[str], ...], None]] = []
spos: t.Optional[int] = None
def _fetch(c: "te.Deque[V]") -> t.Optional[V]:
try:
if spos is None:
return c.popleft()
else:
return c.pop()
except IndexError:
return None
while nargs_spec:
nargs = _fetch(nargs_spec)
if nargs is None:
continue
if nargs == 1:
rv.append(_fetch(args))
elif nargs > 1:
x = [_fetch(args) for _ in range(nargs)]
# If we're reversed, we're pulling in the arguments in reverse,
# so we need to turn them around.
if spos is not None:
x.reverse()
rv.append(tuple(x))
elif nargs < 0:
if spos is not None:
raise TypeError("Cannot have two nargs < 0")
spos = len(rv)
rv.append(None)
# spos is the position of the wildcard (star). If it's not `None`,
# we fill it with the remainder.
if spos is not None:
rv[spos] = tuple(args)
args = []
rv[spos + 1 :] = reversed(rv[spos + 1 :])
return tuple(rv), list(args)
def split_opt(opt: str) -> t.Tuple[str, str]:
first = opt[:1]
if first.isalnum():
return "", opt
if opt[1:2] == first:
return opt[:2], opt[2:]
return first, opt[1:]
def normalize_opt(opt: str, ctx: t.Optional["Context"]) -> str:
if ctx is None or ctx.token_normalize_func is None:
return opt
prefix, opt = split_opt(opt)
return f"{prefix}{ctx.token_normalize_func(opt)}"
def split_arg_string(string: str) -> t.List[str]:
"""Split an argument string as with :func:`shlex.split`, but don't
fail if the string is incomplete. Ignores a missing closing quote or
incomplete escape sequence and uses the partial token as-is.
.. code-block:: python
split_arg_string("example 'my file")
["example", "my file"]
split_arg_string("example my\\")
["example", "my"]
:param string: String to split.
"""
import shlex
lex = shlex.shlex(string, posix=True)
lex.whitespace_split = True
lex.commenters = ""
out = []
try:
for token in lex:
out.append(token)
except ValueError:
# Raised when end-of-string is reached in an invalid state. Use
# the partial token as-is. The quote or escape character is in
# lex.state, not lex.token.
out.append(lex.token)
return out
class Option:
def __init__(
self,
obj: "CoreOption",
opts: t.Sequence[str],
dest: t.Optional[str],
action: t.Optional[str] = None,
nargs: int = 1,
const: t.Optional[t.Any] = None,
):
self._short_opts = []
self._long_opts = []
self.prefixes = set()
for opt in opts:
prefix, value = split_opt(opt)
if not prefix:
raise ValueError(f"Invalid start character for option ({opt})")
self.prefixes.add(prefix[0])
if len(prefix) == 1 and len(value) == 1:
self._short_opts.append(opt)
else:
self._long_opts.append(opt)
self.prefixes.add(prefix)
if action is None:
action = "store"
self.dest = dest
self.action = action
self.nargs = nargs
self.const = const
self.obj = obj
@property
def takes_value(self) -> bool:
return self.action in ("store", "append")
def process(self, value: str, state: "ParsingState") -> None:
if self.action == "store":
state.opts[self.dest] = value # type: ignore
elif self.action == "store_const":
state.opts[self.dest] = self.const # type: ignore
elif self.action == "append":
state.opts.setdefault(self.dest, []).append(value) # type: ignore
elif self.action == "append_const":
state.opts.setdefault(self.dest, []).append(self.const) # type: ignore
elif self.action == "count":
state.opts[self.dest] = state.opts.get(self.dest, 0) + 1 # type: ignore
else:
raise ValueError(f"unknown action '{self.action}'")
state.order.append(self.obj)
class Argument:
def __init__(self, obj: "CoreArgument", dest: t.Optional[str], nargs: int = 1):
self.dest = dest
self.nargs = nargs
self.obj = obj
def process(
self,
value: t.Union[t.Optional[str], t.Sequence[t.Optional[str]]],
state: "ParsingState",
) -> None:
if self.nargs > 1:
assert value is not None
holes = sum(1 for x in value if x is None)
if holes == len(value):
value = None
elif holes != 0:
raise BadArgumentUsage(
_("Argument {name!r} takes {nargs} values.").format(
name=self.dest, nargs=self.nargs
)
)
if self.nargs == -1 and self.obj.envvar is not None and value == ():
# Replace empty tuple with None so that a value from the
# environment may be tried.
value = None
state.opts[self.dest] = value # type: ignore
state.order.append(self.obj)
class ParsingState:
def __init__(self, rargs: t.List[str]) -> None:
self.opts: t.Dict[str, t.Any] = {}
self.largs: t.List[str] = []
self.rargs = rargs
self.order: t.List["CoreParameter"] = []
class OptionParser:
"""The option parser is an internal class that is ultimately used to
parse options and arguments. It's modelled after optparse and brings
a similar but vastly simplified API. It should generally not be used
directly as the high level Click classes wrap it for you.
It's not nearly as extensible as optparse or argparse as it does not
implement features that are implemented on a higher level (such as
types or defaults).
:param ctx: optionally the :class:`~click.Context` where this parser
should go with.
"""
def __init__(self, ctx: t.Optional["Context"] = None) -> None:
#: The :class:`~click.Context` for this parser. This might be
#: `None` for some advanced use cases.
self.ctx = ctx
#: This controls how the parser deals with interspersed arguments.
#: If this is set to `False`, the parser will stop on the first
#: non-option. Click uses this to implement nested subcommands
#: safely.
self.allow_interspersed_args = True
#: This tells the parser how to deal with unknown options. By
#: default it will error out (which is sensible), but there is a
#: second mode where it will ignore it and continue processing
#: after shifting all the unknown options into the resulting args.
self.ignore_unknown_options = False
if ctx is not None:
self.allow_interspersed_args = ctx.allow_interspersed_args
self.ignore_unknown_options = ctx.ignore_unknown_options
self._short_opt: t.Dict[str, Option] = {}
self._long_opt: t.Dict[str, Option] = {}
self._opt_prefixes = {"-", "--"}
self._args: t.List[Argument] = []
def add_option(
self,
obj: "CoreOption",
opts: t.Sequence[str],
dest: t.Optional[str],
action: t.Optional[str] = None,
nargs: int = 1,
const: t.Optional[t.Any] = None,
) -> None:
"""Adds a new option named `dest` to the parser. The destination
is not inferred (unlike with optparse) and needs to be explicitly
provided. Action can be any of ``store``, ``store_const``,
``append``, ``append_const`` or ``count``.
The `obj` can be used to identify the option in the order list
that is returned from the parser.
"""
opts = [normalize_opt(opt, self.ctx) for opt in opts]
option = Option(obj, opts, dest, action=action, nargs=nargs, const=const)
self._opt_prefixes.update(option.prefixes)
for opt in option._short_opts:
self._short_opt[opt] = option
for opt in option._long_opts:
self._long_opt[opt] = option
def add_argument(
self, obj: "CoreArgument", dest: t.Optional[str], nargs: int = 1
) -> None:
"""Adds a positional argument named `dest` to the parser.
The `obj` can be used to identify the option in the order list
that is returned from the parser.
"""
self._args.append(Argument(obj, dest=dest, nargs=nargs))
def parse_args(
self, args: t.List[str]
) -> t.Tuple[t.Dict[str, t.Any], t.List[str], t.List["CoreParameter"]]:
"""Parses positional arguments and returns ``(values, args, order)``
for the parsed options and arguments as well as the leftover
arguments if there are any. The order is a list of objects as they
appear on the command line. If arguments appear multiple times they
will be memorized multiple times as well.
"""
state = ParsingState(args)
try:
self._process_args_for_options(state)
self._process_args_for_args(state)
except UsageError:
if self.ctx is None or not self.ctx.resilient_parsing:
raise
return state.opts, state.largs, state.order
def _process_args_for_args(self, state: ParsingState) -> None:
pargs, args = _unpack_args(
state.largs + state.rargs, [x.nargs for x in self._args]
)
for idx, arg in enumerate(self._args):
arg.process(pargs[idx], state)
state.largs = args
state.rargs = []
def _process_args_for_options(self, state: ParsingState) -> None:
while state.rargs:
arg = state.rargs.pop(0)
arglen = len(arg)
# Double dashes always handled explicitly regardless of what
# prefixes are valid.
if arg == "--":
return
elif arg[:1] in self._opt_prefixes and arglen > 1:
self._process_opts(arg, state)
elif self.allow_interspersed_args:
state.largs.append(arg)
else:
state.rargs.insert(0, arg)
return
# Say this is the original argument list:
# [arg0, arg1, ..., arg(i-1), arg(i), arg(i+1), ..., arg(N-1)]
# ^
# (we are about to process arg(i)).
#
# Then rargs is [arg(i), ..., arg(N-1)] and largs is a *subset* of
# [arg0, ..., arg(i-1)] (any options and their arguments will have
# been removed from largs).
#
# The while loop will usually consume 1 or more arguments per pass.
# If it consumes 1 (eg. arg is an option that takes no arguments),
# then after _process_arg() is done the situation is:
#
# largs = subset of [arg0, ..., arg(i)]
# rargs = [arg(i+1), ..., arg(N-1)]
#
# If allow_interspersed_args is false, largs will always be
# *empty* -- still a subset of [arg0, ..., arg(i-1)], but
# not a very interesting subset!
def _match_long_opt(
self, opt: str, explicit_value: t.Optional[str], state: ParsingState
) -> None:
if opt not in self._long_opt:
from difflib import get_close_matches
possibilities = get_close_matches(opt, self._long_opt)
raise NoSuchOption(opt, possibilities=possibilities, ctx=self.ctx)
option = self._long_opt[opt]
if option.takes_value:
# At this point it's safe to modify rargs by injecting the
# explicit value, because no exception is raised in this
# branch. This means that the inserted value will be fully
# consumed.
if explicit_value is not None:
state.rargs.insert(0, explicit_value)
value = self._get_value_from_state(opt, option, state)
elif explicit_value is not None:
raise BadOptionUsage(
opt, _("Option {name!r} does not take a value.").format(name=opt)
)
else:
value = None
option.process(value, state)
def _match_short_opt(self, arg: str, state: ParsingState) -> None:
stop = False
i = 1
prefix = arg[0]
unknown_options = []
for ch in arg[1:]:
opt = normalize_opt(f"{prefix}{ch}", self.ctx)
option = self._short_opt.get(opt)
i += 1
if not option:
if self.ignore_unknown_options:
unknown_options.append(ch)
continue
raise NoSuchOption(opt, ctx=self.ctx)
if option.takes_value:
# Any characters left in arg? Pretend they're the
# next arg, and stop consuming characters of arg.
if i < len(arg):
state.rargs.insert(0, arg[i:])
stop = True
value = self._get_value_from_state(opt, option, state)
else:
value = None
option.process(value, state)
if stop:
break
# If we got any unknown options we re-combinate the string of the
# remaining options and re-attach the prefix, then report that
# to the state as new larg. This way there is basic combinatorics
# that can be achieved while still ignoring unknown arguments.
if self.ignore_unknown_options and unknown_options:
state.largs.append(f"{prefix}{''.join(unknown_options)}")
def _get_value_from_state(
self, option_name: str, option: Option, state: ParsingState
) -> t.Any:
nargs = option.nargs
if len(state.rargs) < nargs:
if option.obj._flag_needs_value:
# Option allows omitting the value.
value = _flag_needs_value
else:
raise BadOptionUsage(
option_name,
ngettext(
"Option {name!r} requires an argument.",
"Option {name!r} requires {nargs} arguments.",
nargs,
).format(name=option_name, nargs=nargs),
)
elif nargs == 1:
next_rarg = state.rargs[0]
if (
option.obj._flag_needs_value
and isinstance(next_rarg, str)
and next_rarg[:1] in self._opt_prefixes
and len(next_rarg) > 1
):
# The next arg looks like the start of an option, don't
# use it as the value if omitting the value is allowed.
value = _flag_needs_value
else:
value = state.rargs.pop(0)
else:
value = tuple(state.rargs[:nargs])
del state.rargs[:nargs]
return value
def _process_opts(self, arg: str, state: ParsingState) -> None:
explicit_value = None
# Long option handling happens in two parts. The first part is
# supporting explicitly attached values. In any case, we will try
# to long match the option first.
if "=" in arg:
long_opt, explicit_value = arg.split("=", 1)
else:
long_opt = arg
norm_long_opt = normalize_opt(long_opt, self.ctx)
# At this point we will match the (assumed) long option through
# the long option matching code. Note that this allows options
# like "-foo" to be matched as long options.
try:
self._match_long_opt(norm_long_opt, explicit_value, state)
except NoSuchOption:
# At this point the long option matching failed, and we need
# to try with short options. However there is a special rule
# which says, that if we have a two character options prefix
# (applies to "--foo" for instance), we do not dispatch to the
# short option code and will instead raise the no option
# error.
if arg[:2] not in self._opt_prefixes:
self._match_short_opt(arg, state)
return
if not self.ignore_unknown_options:
raise
state.largs.append(arg)

View File

@ -0,0 +1,581 @@
import os
import re
import typing as t
from gettext import gettext as _
from .core import Argument
from .core import BaseCommand
from .core import Context
from .core import MultiCommand
from .core import Option
from .core import Parameter
from .core import ParameterSource
from .parser import split_arg_string
from .utils import echo
def shell_complete(
cli: BaseCommand,
ctx_args: t.Dict[str, t.Any],
prog_name: str,
complete_var: str,
instruction: str,
) -> int:
"""Perform shell completion for the given CLI program.
:param cli: Command being called.
:param ctx_args: Extra arguments to pass to
``cli.make_context``.
:param prog_name: Name of the executable in the shell.
:param complete_var: Name of the environment variable that holds
the completion instruction.
:param instruction: Value of ``complete_var`` with the completion
instruction and shell, in the form ``instruction_shell``.
:return: Status code to exit with.
"""
shell, _, instruction = instruction.partition("_")
comp_cls = get_completion_class(shell)
if comp_cls is None:
return 1
comp = comp_cls(cli, ctx_args, prog_name, complete_var)
if instruction == "source":
echo(comp.source())
return 0
if instruction == "complete":
echo(comp.complete())
return 0
return 1
class CompletionItem:
"""Represents a completion value and metadata about the value. The
default metadata is ``type`` to indicate special shell handling,
and ``help`` if a shell supports showing a help string next to the
value.
Arbitrary parameters can be passed when creating the object, and
accessed using ``item.attr``. If an attribute wasn't passed,
accessing it returns ``None``.
:param value: The completion suggestion.
:param type: Tells the shell script to provide special completion
support for the type. Click uses ``"dir"`` and ``"file"``.
:param help: String shown next to the value if supported.
:param kwargs: Arbitrary metadata. The built-in implementations
don't use this, but custom type completions paired with custom
shell support could use it.
"""
__slots__ = ("value", "type", "help", "_info")
def __init__(
self,
value: t.Any,
type: str = "plain",
help: t.Optional[str] = None,
**kwargs: t.Any,
) -> None:
self.value = value
self.type = type
self.help = help
self._info = kwargs
def __getattr__(self, name: str) -> t.Any:
return self._info.get(name)
# Only Bash >= 4.4 has the nosort option.
_SOURCE_BASH = """\
%(complete_func)s() {
local IFS=$'\\n'
local response
response=$(env COMP_WORDS="${COMP_WORDS[*]}" COMP_CWORD=$COMP_CWORD \
%(complete_var)s=bash_complete $1)
for completion in $response; do
IFS=',' read type value <<< "$completion"
if [[ $type == 'dir' ]]; then
COMREPLY=()
compopt -o dirnames
elif [[ $type == 'file' ]]; then
COMREPLY=()
compopt -o default
elif [[ $type == 'plain' ]]; then
COMPREPLY+=($value)
fi
done
return 0
}
%(complete_func)s_setup() {
complete -o nosort -F %(complete_func)s %(prog_name)s
}
%(complete_func)s_setup;
"""
_SOURCE_ZSH = """\
#compdef %(prog_name)s
%(complete_func)s() {
local -a completions
local -a completions_with_descriptions
local -a response
(( ! $+commands[%(prog_name)s] )) && return 1
response=("${(@f)$(env COMP_WORDS="${words[*]}" COMP_CWORD=$((CURRENT-1)) \
%(complete_var)s=zsh_complete %(prog_name)s)}")
for type key descr in ${response}; do
if [[ "$type" == "plain" ]]; then
if [[ "$descr" == "_" ]]; then
completions+=("$key")
else
completions_with_descriptions+=("$key":"$descr")
fi
elif [[ "$type" == "dir" ]]; then
_path_files -/
elif [[ "$type" == "file" ]]; then
_path_files -f
fi
done
if [ -n "$completions_with_descriptions" ]; then
_describe -V unsorted completions_with_descriptions -U
fi
if [ -n "$completions" ]; then
compadd -U -V unsorted -a completions
fi
}
compdef %(complete_func)s %(prog_name)s;
"""
_SOURCE_FISH = """\
function %(complete_func)s;
set -l response;
for value in (env %(complete_var)s=fish_complete COMP_WORDS=(commandline -cp) \
COMP_CWORD=(commandline -t) %(prog_name)s);
set response $response $value;
end;
for completion in $response;
set -l metadata (string split "," $completion);
if test $metadata[1] = "dir";
__fish_complete_directories $metadata[2];
else if test $metadata[1] = "file";
__fish_complete_path $metadata[2];
else if test $metadata[1] = "plain";
echo $metadata[2];
end;
end;
end;
complete --no-files --command %(prog_name)s --arguments \
"(%(complete_func)s)";
"""
class ShellComplete:
"""Base class for providing shell completion support. A subclass for
a given shell will override attributes and methods to implement the
completion instructions (``source`` and ``complete``).
:param cli: Command being called.
:param prog_name: Name of the executable in the shell.
:param complete_var: Name of the environment variable that holds
the completion instruction.
.. versionadded:: 8.0
"""
name: t.ClassVar[str]
"""Name to register the shell as with :func:`add_completion_class`.
This is used in completion instructions (``{name}_source`` and
``{name}_complete``).
"""
source_template: t.ClassVar[str]
"""Completion script template formatted by :meth:`source`. This must
be provided by subclasses.
"""
def __init__(
self,
cli: BaseCommand,
ctx_args: t.Dict[str, t.Any],
prog_name: str,
complete_var: str,
) -> None:
self.cli = cli
self.ctx_args = ctx_args
self.prog_name = prog_name
self.complete_var = complete_var
@property
def func_name(self) -> str:
"""The name of the shell function defined by the completion
script.
"""
safe_name = re.sub(r"\W*", "", self.prog_name.replace("-", "_"), re.ASCII)
return f"_{safe_name}_completion"
def source_vars(self) -> t.Dict[str, t.Any]:
"""Vars for formatting :attr:`source_template`.
By default this provides ``complete_func``, ``complete_var``,
and ``prog_name``.
"""
return {
"complete_func": self.func_name,
"complete_var": self.complete_var,
"prog_name": self.prog_name,
}
def source(self) -> str:
"""Produce the shell script that defines the completion
function. By default this ``%``-style formats
:attr:`source_template` with the dict returned by
:meth:`source_vars`.
"""
return self.source_template % self.source_vars()
def get_completion_args(self) -> t.Tuple[t.List[str], str]:
"""Use the env vars defined by the shell script to return a
tuple of ``args, incomplete``. This must be implemented by
subclasses.
"""
raise NotImplementedError
def get_completions(
self, args: t.List[str], incomplete: str
) -> t.List[CompletionItem]:
"""Determine the context and last complete command or parameter
from the complete args. Call that object's ``shell_complete``
method to get the completions for the incomplete value.
:param args: List of complete args before the incomplete value.
:param incomplete: Value being completed. May be empty.
"""
ctx = _resolve_context(self.cli, self.ctx_args, self.prog_name, args)
obj, incomplete = _resolve_incomplete(ctx, args, incomplete)
return obj.shell_complete(ctx, incomplete)
def format_completion(self, item: CompletionItem) -> str:
"""Format a completion item into the form recognized by the
shell script. This must be implemented by subclasses.
:param item: Completion item to format.
"""
raise NotImplementedError
def complete(self) -> str:
"""Produce the completion data to send back to the shell.
By default this calls :meth:`get_completion_args`, gets the
completions, then calls :meth:`format_completion` for each
completion.
"""
args, incomplete = self.get_completion_args()
completions = self.get_completions(args, incomplete)
out = [self.format_completion(item) for item in completions]
return "\n".join(out)
class BashComplete(ShellComplete):
"""Shell completion for Bash."""
name = "bash"
source_template = _SOURCE_BASH
def _check_version(self) -> None:
import subprocess
output = subprocess.run(
["bash", "-c", "echo ${BASH_VERSION}"], stdout=subprocess.PIPE
)
match = re.search(r"^(\d+)\.(\d+)\.\d+", output.stdout.decode())
if match is not None:
major, minor = match.groups()
if major < "4" or major == "4" and minor < "4":
raise RuntimeError(
_(
"Shell completion is not supported for Bash"
" versions older than 4.4."
)
)
else:
raise RuntimeError(
_("Couldn't detect Bash version, shell completion is not supported.")
)
def source(self) -> str:
self._check_version()
return super().source()
def get_completion_args(self) -> t.Tuple[t.List[str], str]:
cwords = split_arg_string(os.environ["COMP_WORDS"])
cword = int(os.environ["COMP_CWORD"])
args = cwords[1:cword]
try:
incomplete = cwords[cword]
except IndexError:
incomplete = ""
return args, incomplete
def format_completion(self, item: CompletionItem) -> str:
return f"{item.type},{item.value}"
class ZshComplete(ShellComplete):
"""Shell completion for Zsh."""
name = "zsh"
source_template = _SOURCE_ZSH
def get_completion_args(self) -> t.Tuple[t.List[str], str]:
cwords = split_arg_string(os.environ["COMP_WORDS"])
cword = int(os.environ["COMP_CWORD"])
args = cwords[1:cword]
try:
incomplete = cwords[cword]
except IndexError:
incomplete = ""
return args, incomplete
def format_completion(self, item: CompletionItem) -> str:
return f"{item.type}\n{item.value}\n{item.help if item.help else '_'}"
class FishComplete(ShellComplete):
"""Shell completion for Fish."""
name = "fish"
source_template = _SOURCE_FISH
def get_completion_args(self) -> t.Tuple[t.List[str], str]:
cwords = split_arg_string(os.environ["COMP_WORDS"])
incomplete = os.environ["COMP_CWORD"]
args = cwords[1:]
# Fish stores the partial word in both COMP_WORDS and
# COMP_CWORD, remove it from complete args.
if incomplete and args and args[-1] == incomplete:
args.pop()
return args, incomplete
def format_completion(self, item: CompletionItem) -> str:
if item.help:
return f"{item.type},{item.value}\t{item.help}"
return f"{item.type},{item.value}"
_available_shells: t.Dict[str, t.Type[ShellComplete]] = {
"bash": BashComplete,
"fish": FishComplete,
"zsh": ZshComplete,
}
def add_completion_class(
cls: t.Type[ShellComplete], name: t.Optional[str] = None
) -> None:
"""Register a :class:`ShellComplete` subclass under the given name.
The name will be provided by the completion instruction environment
variable during completion.
:param cls: The completion class that will handle completion for the
shell.
:param name: Name to register the class under. Defaults to the
class's ``name`` attribute.
"""
if name is None:
name = cls.name
_available_shells[name] = cls
def get_completion_class(shell: str) -> t.Optional[t.Type[ShellComplete]]:
"""Look up a registered :class:`ShellComplete` subclass by the name
provided by the completion instruction environment variable. If the
name isn't registered, returns ``None``.
:param shell: Name the class is registered under.
"""
return _available_shells.get(shell)
def _is_incomplete_argument(ctx: Context, param: Parameter) -> bool:
"""Determine if the given parameter is an argument that can still
accept values.
:param ctx: Invocation context for the command represented by the
parsed complete args.
:param param: Argument object being checked.
"""
if not isinstance(param, Argument):
return False
assert param.name is not None
value = ctx.params[param.name]
return (
param.nargs == -1
or ctx.get_parameter_source(param.name) is not ParameterSource.COMMANDLINE
or (
param.nargs > 1
and isinstance(value, (tuple, list))
and len(value) < param.nargs
)
)
def _start_of_option(value: str) -> bool:
"""Check if the value looks like the start of an option."""
if not value:
return False
c = value[0]
# Allow "/" since that starts a path.
return not c.isalnum() and c != "/"
def _is_incomplete_option(args: t.List[str], param: Parameter) -> bool:
"""Determine if the given parameter is an option that needs a value.
:param args: List of complete args before the incomplete value.
:param param: Option object being checked.
"""
if not isinstance(param, Option):
return False
if param.is_flag:
return False
last_option = None
for index, arg in enumerate(reversed(args)):
if index + 1 > param.nargs:
break
if _start_of_option(arg):
last_option = arg
return last_option is not None and last_option in param.opts
def _resolve_context(
cli: BaseCommand, ctx_args: t.Dict[str, t.Any], prog_name: str, args: t.List[str]
) -> Context:
"""Produce the context hierarchy starting with the command and
traversing the complete arguments. This only follows the commands,
it doesn't trigger input prompts or callbacks.
:param cli: Command being called.
:param prog_name: Name of the executable in the shell.
:param args: List of complete args before the incomplete value.
"""
ctx_args["resilient_parsing"] = True
ctx = cli.make_context(prog_name, args.copy(), **ctx_args)
args = ctx.protected_args + ctx.args
while args:
command = ctx.command
if isinstance(command, MultiCommand):
if not command.chain:
name, cmd, args = command.resolve_command(ctx, args)
if cmd is None:
return ctx
ctx = cmd.make_context(name, args, parent=ctx, resilient_parsing=True)
args = ctx.protected_args + ctx.args
else:
while args:
name, cmd, args = command.resolve_command(ctx, args)
if cmd is None:
return ctx
sub_ctx = cmd.make_context(
name,
args,
parent=ctx,
allow_extra_args=True,
allow_interspersed_args=False,
resilient_parsing=True,
)
args = sub_ctx.args
ctx = sub_ctx
args = [*sub_ctx.protected_args, *sub_ctx.args]
else:
break
return ctx
def _resolve_incomplete(
ctx: Context, args: t.List[str], incomplete: str
) -> t.Tuple[t.Union[BaseCommand, Parameter], str]:
"""Find the Click object that will handle the completion of the
incomplete value. Return the object and the incomplete value.
:param ctx: Invocation context for the command represented by
the parsed complete args.
:param args: List of complete args before the incomplete value.
:param incomplete: Value being completed. May be empty.
"""
# Different shells treat an "=" between a long option name and
# value differently. Might keep the value joined, return the "="
# as a separate item, or return the split name and value. Always
# split and discard the "=" to make completion easier.
if incomplete == "=":
incomplete = ""
elif "=" in incomplete and _start_of_option(incomplete):
name, _, incomplete = incomplete.partition("=")
args.append(name)
# The "--" marker tells Click to stop treating values as options
# even if they start with the option character. If it hasn't been
# given and the incomplete arg looks like an option, the current
# command will provide option name completions.
if "--" not in args and _start_of_option(incomplete):
return ctx.command, incomplete
params = ctx.command.get_params(ctx)
# If the last complete arg is an option name with an incomplete
# value, the option will provide value completions.
for param in params:
if _is_incomplete_option(args, param):
return param, incomplete
# It's not an option name or value. The first argument without a
# parsed value will provide value completions.
for param in params:
if _is_incomplete_argument(ctx, param):
return param, incomplete
# There were no unparsed arguments, the command may be a group that
# will provide command name completions.
return ctx.command, incomplete

View File

@ -0,0 +1,809 @@
import inspect
import io
import itertools
import os
import sys
import typing
import typing as t
from gettext import gettext as _
from ._compat import isatty
from ._compat import strip_ansi
from ._compat import WIN
from .exceptions import Abort
from .exceptions import UsageError
from .globals import resolve_color_default
from .types import Choice
from .types import convert_type
from .types import ParamType
from .utils import echo
from .utils import LazyFile
if t.TYPE_CHECKING:
from ._termui_impl import ProgressBar
V = t.TypeVar("V")
# The prompt functions to use. The doc tools currently override these
# functions to customize how they work.
visible_prompt_func: t.Callable[[str], str] = input
_ansi_colors = {
"black": 30,
"red": 31,
"green": 32,
"yellow": 33,
"blue": 34,
"magenta": 35,
"cyan": 36,
"white": 37,
"reset": 39,
"bright_black": 90,
"bright_red": 91,
"bright_green": 92,
"bright_yellow": 93,
"bright_blue": 94,
"bright_magenta": 95,
"bright_cyan": 96,
"bright_white": 97,
}
_ansi_reset_all = "\033[0m"
def hidden_prompt_func(prompt: str) -> str:
import getpass
return getpass.getpass(prompt)
def _build_prompt(
text: str,
suffix: str,
show_default: bool = False,
default: t.Optional[t.Any] = None,
show_choices: bool = True,
type: t.Optional[ParamType] = None,
) -> str:
prompt = text
if type is not None and show_choices and isinstance(type, Choice):
prompt += f" ({', '.join(map(str, type.choices))})"
if default is not None and show_default:
prompt = f"{prompt} [{_format_default(default)}]"
return f"{prompt}{suffix}"
def _format_default(default: t.Any) -> t.Any:
if isinstance(default, (io.IOBase, LazyFile)) and hasattr(default, "name"):
return default.name # type: ignore
return default
def prompt(
text: str,
default: t.Optional[t.Any] = None,
hide_input: bool = False,
confirmation_prompt: t.Union[bool, str] = False,
type: t.Optional[t.Union[ParamType, t.Any]] = None,
value_proc: t.Optional[t.Callable[[str], t.Any]] = None,
prompt_suffix: str = ": ",
show_default: bool = True,
err: bool = False,
show_choices: bool = True,
) -> t.Any:
"""Prompts a user for input. This is a convenience function that can
be used to prompt a user for input later.
If the user aborts the input by sending a interrupt signal, this
function will catch it and raise a :exc:`Abort` exception.
:param text: the text to show for the prompt.
:param default: the default value to use if no input happens. If this
is not given it will prompt until it's aborted.
:param hide_input: if this is set to true then the input value will
be hidden.
:param confirmation_prompt: Prompt a second time to confirm the
value. Can be set to a string instead of ``True`` to customize
the message.
:param type: the type to use to check the value against.
:param value_proc: if this parameter is provided it's a function that
is invoked instead of the type conversion to
convert a value.
:param prompt_suffix: a suffix that should be added to the prompt.
:param show_default: shows or hides the default value in the prompt.
:param err: if set to true the file defaults to ``stderr`` instead of
``stdout``, the same as with echo.
:param show_choices: Show or hide choices if the passed type is a Choice.
For example if type is a Choice of either day or week,
show_choices is true and text is "Group by" then the
prompt will be "Group by (day, week): ".
.. versionadded:: 8.0
``confirmation_prompt`` can be a custom string.
.. versionadded:: 7.0
Added the ``show_choices`` parameter.
.. versionadded:: 6.0
Added unicode support for cmd.exe on Windows.
.. versionadded:: 4.0
Added the `err` parameter.
"""
def prompt_func(text: str) -> str:
f = hidden_prompt_func if hide_input else visible_prompt_func
try:
# Write the prompt separately so that we get nice
# coloring through colorama on Windows
echo(text.rstrip(" "), nl=False, err=err)
# Echo a space to stdout to work around an issue where
# readline causes backspace to clear the whole line.
return f(" ")
except (KeyboardInterrupt, EOFError):
# getpass doesn't print a newline if the user aborts input with ^C.
# Allegedly this behavior is inherited from getpass(3).
# A doc bug has been filed at https://bugs.python.org/issue24711
if hide_input:
echo(None, err=err)
raise Abort() from None
if value_proc is None:
value_proc = convert_type(type, default)
prompt = _build_prompt(
text, prompt_suffix, show_default, default, show_choices, type
)
if confirmation_prompt:
if confirmation_prompt is True:
confirmation_prompt = _("Repeat for confirmation")
confirmation_prompt = t.cast(str, confirmation_prompt)
confirmation_prompt = _build_prompt(confirmation_prompt, prompt_suffix)
while True:
while True:
value = prompt_func(prompt)
if value:
break
elif default is not None:
value = default
break
try:
result = value_proc(value)
except UsageError as e:
if hide_input:
echo(_("Error: The value you entered was invalid."), err=err)
else:
echo(_("Error: {e.message}").format(e=e), err=err) # noqa: B306
continue
if not confirmation_prompt:
return result
while True:
confirmation_prompt = t.cast(str, confirmation_prompt)
value2 = prompt_func(confirmation_prompt)
if value2:
break
if value == value2:
return result
echo(_("Error: The two entered values do not match."), err=err)
def confirm(
text: str,
default: t.Optional[bool] = False,
abort: bool = False,
prompt_suffix: str = ": ",
show_default: bool = True,
err: bool = False,
) -> bool:
"""Prompts for confirmation (yes/no question).
If the user aborts the input by sending a interrupt signal this
function will catch it and raise a :exc:`Abort` exception.
:param text: the question to ask.
:param default: The default value to use when no input is given. If
``None``, repeat until input is given.
:param abort: if this is set to `True` a negative answer aborts the
exception by raising :exc:`Abort`.
:param prompt_suffix: a suffix that should be added to the prompt.
:param show_default: shows or hides the default value in the prompt.
:param err: if set to true the file defaults to ``stderr`` instead of
``stdout``, the same as with echo.
.. versionchanged:: 8.0
Repeat until input is given if ``default`` is ``None``.
.. versionadded:: 4.0
Added the ``err`` parameter.
"""
prompt = _build_prompt(
text,
prompt_suffix,
show_default,
"y/n" if default is None else ("Y/n" if default else "y/N"),
)
while True:
try:
# Write the prompt separately so that we get nice
# coloring through colorama on Windows
echo(prompt.rstrip(" "), nl=False, err=err)
# Echo a space to stdout to work around an issue where
# readline causes backspace to clear the whole line.
value = visible_prompt_func(" ").lower().strip()
except (KeyboardInterrupt, EOFError):
raise Abort() from None
if value in ("y", "yes"):
rv = True
elif value in ("n", "no"):
rv = False
elif default is not None and value == "":
rv = default
else:
echo(_("Error: invalid input"), err=err)
continue
break
if abort and not rv:
raise Abort()
return rv
def get_terminal_size() -> os.terminal_size:
"""Returns the current size of the terminal as tuple in the form
``(width, height)`` in columns and rows.
.. deprecated:: 8.0
Will be removed in Click 8.1. Use
:func:`shutil.get_terminal_size` instead.
"""
import shutil
import warnings
warnings.warn(
"'click.get_terminal_size()' is deprecated and will be removed"
" in Click 8.1. Use 'shutil.get_terminal_size()' instead.",
DeprecationWarning,
stacklevel=2,
)
return shutil.get_terminal_size()
def echo_via_pager(
text_or_generator: t.Union[t.Iterable[str], t.Callable[[], t.Iterable[str]], str],
color: t.Optional[bool] = None,
) -> None:
"""This function takes a text and shows it via an environment specific
pager on stdout.
.. versionchanged:: 3.0
Added the `color` flag.
:param text_or_generator: the text to page, or alternatively, a
generator emitting the text to page.
:param color: controls if the pager supports ANSI colors or not. The
default is autodetection.
"""
color = resolve_color_default(color)
if inspect.isgeneratorfunction(text_or_generator):
i = t.cast(t.Callable[[], t.Iterable[str]], text_or_generator)()
elif isinstance(text_or_generator, str):
i = [text_or_generator]
else:
i = iter(t.cast(t.Iterable[str], text_or_generator))
# convert every element of i to a text type if necessary
text_generator = (el if isinstance(el, str) else str(el) for el in i)
from ._termui_impl import pager
return pager(itertools.chain(text_generator, "\n"), color)
def progressbar(
iterable: t.Optional[t.Iterable[V]] = None,
length: t.Optional[int] = None,
label: t.Optional[str] = None,
show_eta: bool = True,
show_percent: t.Optional[bool] = None,
show_pos: bool = False,
item_show_func: t.Optional[t.Callable[[t.Optional[V]], t.Optional[str]]] = None,
fill_char: str = "#",
empty_char: str = "-",
bar_template: str = "%(label)s [%(bar)s] %(info)s",
info_sep: str = " ",
width: int = 36,
file: t.Optional[t.TextIO] = None,
color: t.Optional[bool] = None,
update_min_steps: int = 1,
) -> "ProgressBar[V]":
"""This function creates an iterable context manager that can be used
to iterate over something while showing a progress bar. It will
either iterate over the `iterable` or `length` items (that are counted
up). While iteration happens, this function will print a rendered
progress bar to the given `file` (defaults to stdout) and will attempt
to calculate remaining time and more. By default, this progress bar
will not be rendered if the file is not a terminal.
The context manager creates the progress bar. When the context
manager is entered the progress bar is already created. With every
iteration over the progress bar, the iterable passed to the bar is
advanced and the bar is updated. When the context manager exits,
a newline is printed and the progress bar is finalized on screen.
Note: The progress bar is currently designed for use cases where the
total progress can be expected to take at least several seconds.
Because of this, the ProgressBar class object won't display
progress that is considered too fast, and progress where the time
between steps is less than a second.
No printing must happen or the progress bar will be unintentionally
destroyed.
Example usage::
with progressbar(items) as bar:
for item in bar:
do_something_with(item)
Alternatively, if no iterable is specified, one can manually update the
progress bar through the `update()` method instead of directly
iterating over the progress bar. The update method accepts the number
of steps to increment the bar with::
with progressbar(length=chunks.total_bytes) as bar:
for chunk in chunks:
process_chunk(chunk)
bar.update(chunks.bytes)
The ``update()`` method also takes an optional value specifying the
``current_item`` at the new position. This is useful when used
together with ``item_show_func`` to customize the output for each
manual step::
with click.progressbar(
length=total_size,
label='Unzipping archive',
item_show_func=lambda a: a.filename
) as bar:
for archive in zip_file:
archive.extract()
bar.update(archive.size, archive)
:param iterable: an iterable to iterate over. If not provided the length
is required.
:param length: the number of items to iterate over. By default the
progressbar will attempt to ask the iterator about its
length, which might or might not work. If an iterable is
also provided this parameter can be used to override the
length. If an iterable is not provided the progress bar
will iterate over a range of that length.
:param label: the label to show next to the progress bar.
:param show_eta: enables or disables the estimated time display. This is
automatically disabled if the length cannot be
determined.
:param show_percent: enables or disables the percentage display. The
default is `True` if the iterable has a length or
`False` if not.
:param show_pos: enables or disables the absolute position display. The
default is `False`.
:param item_show_func: A function called with the current item which
can return a string to show next to the progress bar. If the
function returns ``None`` nothing is shown. The current item can
be ``None``, such as when entering and exiting the bar.
:param fill_char: the character to use to show the filled part of the
progress bar.
:param empty_char: the character to use to show the non-filled part of
the progress bar.
:param bar_template: the format string to use as template for the bar.
The parameters in it are ``label`` for the label,
``bar`` for the progress bar and ``info`` for the
info section.
:param info_sep: the separator between multiple info items (eta etc.)
:param width: the width of the progress bar in characters, 0 means full
terminal width
:param file: The file to write to. If this is not a terminal then
only the label is printed.
:param color: controls if the terminal supports ANSI colors or not. The
default is autodetection. This is only needed if ANSI
codes are included anywhere in the progress bar output
which is not the case by default.
:param update_min_steps: Render only when this many updates have
completed. This allows tuning for very fast iterators.
.. versionchanged:: 8.0
Output is shown even if execution time is less than 0.5 seconds.
.. versionchanged:: 8.0
``item_show_func`` shows the current item, not the previous one.
.. versionchanged:: 8.0
Labels are echoed if the output is not a TTY. Reverts a change
in 7.0 that removed all output.
.. versionadded:: 8.0
Added the ``update_min_steps`` parameter.
.. versionchanged:: 4.0
Added the ``color`` parameter. Added the ``update`` method to
the object.
.. versionadded:: 2.0
"""
from ._termui_impl import ProgressBar
color = resolve_color_default(color)
return ProgressBar(
iterable=iterable,
length=length,
show_eta=show_eta,
show_percent=show_percent,
show_pos=show_pos,
item_show_func=item_show_func,
fill_char=fill_char,
empty_char=empty_char,
bar_template=bar_template,
info_sep=info_sep,
file=file,
label=label,
width=width,
color=color,
update_min_steps=update_min_steps,
)
def clear() -> None:
"""Clears the terminal screen. This will have the effect of clearing
the whole visible space of the terminal and moving the cursor to the
top left. This does not do anything if not connected to a terminal.
.. versionadded:: 2.0
"""
if not isatty(sys.stdout):
return
if WIN:
os.system("cls")
else:
sys.stdout.write("\033[2J\033[1;1H")
def _interpret_color(
color: t.Union[int, t.Tuple[int, int, int], str], offset: int = 0
) -> str:
if isinstance(color, int):
return f"{38 + offset};5;{color:d}"
if isinstance(color, (tuple, list)):
r, g, b = color
return f"{38 + offset};2;{r:d};{g:d};{b:d}"
return str(_ansi_colors[color] + offset)
def style(
text: t.Any,
fg: t.Optional[t.Union[int, t.Tuple[int, int, int], str]] = None,
bg: t.Optional[t.Union[int, t.Tuple[int, int, int], str]] = None,
bold: t.Optional[bool] = None,
dim: t.Optional[bool] = None,
underline: t.Optional[bool] = None,
overline: t.Optional[bool] = None,
italic: t.Optional[bool] = None,
blink: t.Optional[bool] = None,
reverse: t.Optional[bool] = None,
strikethrough: t.Optional[bool] = None,
reset: bool = True,
) -> str:
"""Styles a text with ANSI styles and returns the new string. By
default the styling is self contained which means that at the end
of the string a reset code is issued. This can be prevented by
passing ``reset=False``.
Examples::
click.echo(click.style('Hello World!', fg='green'))
click.echo(click.style('ATTENTION!', blink=True))
click.echo(click.style('Some things', reverse=True, fg='cyan'))
click.echo(click.style('More colors', fg=(255, 12, 128), bg=117))
Supported color names:
* ``black`` (might be a gray)
* ``red``
* ``green``
* ``yellow`` (might be an orange)
* ``blue``
* ``magenta``
* ``cyan``
* ``white`` (might be light gray)
* ``bright_black``
* ``bright_red``
* ``bright_green``
* ``bright_yellow``
* ``bright_blue``
* ``bright_magenta``
* ``bright_cyan``
* ``bright_white``
* ``reset`` (reset the color code only)
If the terminal supports it, color may also be specified as:
- An integer in the interval [0, 255]. The terminal must support
8-bit/256-color mode.
- An RGB tuple of three integers in [0, 255]. The terminal must
support 24-bit/true-color mode.
See https://en.wikipedia.org/wiki/ANSI_color and
https://gist.github.com/XVilka/8346728 for more information.
:param text: the string to style with ansi codes.
:param fg: if provided this will become the foreground color.
:param bg: if provided this will become the background color.
:param bold: if provided this will enable or disable bold mode.
:param dim: if provided this will enable or disable dim mode. This is
badly supported.
:param underline: if provided this will enable or disable underline.
:param overline: if provided this will enable or disable overline.
:param italic: if provided this will enable or disable italic.
:param blink: if provided this will enable or disable blinking.
:param reverse: if provided this will enable or disable inverse
rendering (foreground becomes background and the
other way round).
:param strikethrough: if provided this will enable or disable
striking through text.
:param reset: by default a reset-all code is added at the end of the
string which means that styles do not carry over. This
can be disabled to compose styles.
.. versionchanged:: 8.0
A non-string ``message`` is converted to a string.
.. versionchanged:: 8.0
Added support for 256 and RGB color codes.
.. versionchanged:: 8.0
Added the ``strikethrough``, ``italic``, and ``overline``
parameters.
.. versionchanged:: 7.0
Added support for bright colors.
.. versionadded:: 2.0
"""
if not isinstance(text, str):
text = str(text)
bits = []
if fg:
try:
bits.append(f"\033[{_interpret_color(fg)}m")
except KeyError:
raise TypeError(f"Unknown color {fg!r}") from None
if bg:
try:
bits.append(f"\033[{_interpret_color(bg, 10)}m")
except KeyError:
raise TypeError(f"Unknown color {bg!r}") from None
if bold is not None:
bits.append(f"\033[{1 if bold else 22}m")
if dim is not None:
bits.append(f"\033[{2 if dim else 22}m")
if underline is not None:
bits.append(f"\033[{4 if underline else 24}m")
if overline is not None:
bits.append(f"\033[{53 if overline else 55}m")
if italic is not None:
bits.append(f"\033[{3 if italic else 23}m")
if blink is not None:
bits.append(f"\033[{5 if blink else 25}m")
if reverse is not None:
bits.append(f"\033[{7 if reverse else 27}m")
if strikethrough is not None:
bits.append(f"\033[{9 if strikethrough else 29}m")
bits.append(text)
if reset:
bits.append(_ansi_reset_all)
return "".join(bits)
def unstyle(text: str) -> str:
"""Removes ANSI styling information from a string. Usually it's not
necessary to use this function as Click's echo function will
automatically remove styling if necessary.
.. versionadded:: 2.0
:param text: the text to remove style information from.
"""
return strip_ansi(text)
def secho(
message: t.Optional[t.Any] = None,
file: t.Optional[t.IO] = None,
nl: bool = True,
err: bool = False,
color: t.Optional[bool] = None,
**styles: t.Any,
) -> None:
"""This function combines :func:`echo` and :func:`style` into one
call. As such the following two calls are the same::
click.secho('Hello World!', fg='green')
click.echo(click.style('Hello World!', fg='green'))
All keyword arguments are forwarded to the underlying functions
depending on which one they go with.
Non-string types will be converted to :class:`str`. However,
:class:`bytes` are passed directly to :meth:`echo` without applying
style. If you want to style bytes that represent text, call
:meth:`bytes.decode` first.
.. versionchanged:: 8.0
A non-string ``message`` is converted to a string. Bytes are
passed through without style applied.
.. versionadded:: 2.0
"""
if message is not None and not isinstance(message, (bytes, bytearray)):
message = style(message, **styles)
return echo(message, file=file, nl=nl, err=err, color=color)
def edit(
text: t.Optional[t.AnyStr] = None,
editor: t.Optional[str] = None,
env: t.Optional[t.Mapping[str, str]] = None,
require_save: bool = True,
extension: str = ".txt",
filename: t.Optional[str] = None,
) -> t.Optional[t.AnyStr]:
r"""Edits the given text in the defined editor. If an editor is given
(should be the full path to the executable but the regular operating
system search path is used for finding the executable) it overrides
the detected editor. Optionally, some environment variables can be
used. If the editor is closed without changes, `None` is returned. In
case a file is edited directly the return value is always `None` and
`require_save` and `extension` are ignored.
If the editor cannot be opened a :exc:`UsageError` is raised.
Note for Windows: to simplify cross-platform usage, the newlines are
automatically converted from POSIX to Windows and vice versa. As such,
the message here will have ``\n`` as newline markers.
:param text: the text to edit.
:param editor: optionally the editor to use. Defaults to automatic
detection.
:param env: environment variables to forward to the editor.
:param require_save: if this is true, then not saving in the editor
will make the return value become `None`.
:param extension: the extension to tell the editor about. This defaults
to `.txt` but changing this might change syntax
highlighting.
:param filename: if provided it will edit this file instead of the
provided text contents. It will not use a temporary
file as an indirection in that case.
"""
from ._termui_impl import Editor
ed = Editor(editor=editor, env=env, require_save=require_save, extension=extension)
if filename is None:
return ed.edit(text)
ed.edit_file(filename)
return None
def launch(url: str, wait: bool = False, locate: bool = False) -> int:
"""This function launches the given URL (or filename) in the default
viewer application for this file type. If this is an executable, it
might launch the executable in a new session. The return value is
the exit code of the launched application. Usually, ``0`` indicates
success.
Examples::
click.launch('https://click.palletsprojects.com/')
click.launch('/my/downloaded/file', locate=True)
.. versionadded:: 2.0
:param url: URL or filename of the thing to launch.
:param wait: Wait for the program to exit before returning. This
only works if the launched program blocks. In particular,
``xdg-open`` on Linux does not block.
:param locate: if this is set to `True` then instead of launching the
application associated with the URL it will attempt to
launch a file manager with the file located. This
might have weird effects if the URL does not point to
the filesystem.
"""
from ._termui_impl import open_url
return open_url(url, wait=wait, locate=locate)
# If this is provided, getchar() calls into this instead. This is used
# for unittesting purposes.
_getchar: t.Optional[t.Callable[[bool], str]] = None
def getchar(echo: bool = False) -> str:
"""Fetches a single character from the terminal and returns it. This
will always return a unicode character and under certain rare
circumstances this might return more than one character. The
situations which more than one character is returned is when for
whatever reason multiple characters end up in the terminal buffer or
standard input was not actually a terminal.
Note that this will always read from the terminal, even if something
is piped into the standard input.
Note for Windows: in rare cases when typing non-ASCII characters, this
function might wait for a second character and then return both at once.
This is because certain Unicode characters look like special-key markers.
.. versionadded:: 2.0
:param echo: if set to `True`, the character read will also show up on
the terminal. The default is to not show it.
"""
global _getchar
if _getchar is None:
from ._termui_impl import getchar as f
_getchar = f
return _getchar(echo)
def raw_terminal() -> t.ContextManager[int]:
from ._termui_impl import raw_terminal as f
return f()
def pause(info: t.Optional[str] = None, err: bool = False) -> None:
"""This command stops execution and waits for the user to press any
key to continue. This is similar to the Windows batch "pause"
command. If the program is not run through a terminal, this command
will instead do nothing.
.. versionadded:: 2.0
.. versionadded:: 4.0
Added the `err` parameter.
:param info: The message to print before pausing. Defaults to
``"Press any key to continue..."``.
:param err: if set to message goes to ``stderr`` instead of
``stdout``, the same as with echo.
"""
if not isatty(sys.stdin) or not isatty(sys.stdout):
return
if info is None:
info = _("Press any key to continue...")
try:
if info:
echo(info, nl=False, err=err)
try:
getchar()
except (KeyboardInterrupt, EOFError):
pass
finally:
if info:
echo(err=err)

View File

@ -0,0 +1,479 @@
import contextlib
import io
import os
import shlex
import shutil
import sys
import tempfile
import typing as t
from types import TracebackType
from . import formatting
from . import termui
from . import utils
from ._compat import _find_binary_reader
if t.TYPE_CHECKING:
from .core import BaseCommand
class EchoingStdin:
def __init__(self, input: t.BinaryIO, output: t.BinaryIO) -> None:
self._input = input
self._output = output
self._paused = False
def __getattr__(self, x: str) -> t.Any:
return getattr(self._input, x)
def _echo(self, rv: bytes) -> bytes:
if not self._paused:
self._output.write(rv)
return rv
def read(self, n: int = -1) -> bytes:
return self._echo(self._input.read(n))
def read1(self, n: int = -1) -> bytes:
return self._echo(self._input.read1(n)) # type: ignore
def readline(self, n: int = -1) -> bytes:
return self._echo(self._input.readline(n))
def readlines(self) -> t.List[bytes]:
return [self._echo(x) for x in self._input.readlines()]
def __iter__(self) -> t.Iterator[bytes]:
return iter(self._echo(x) for x in self._input)
def __repr__(self) -> str:
return repr(self._input)
@contextlib.contextmanager
def _pause_echo(stream: t.Optional[EchoingStdin]) -> t.Iterator[None]:
if stream is None:
yield
else:
stream._paused = True
yield
stream._paused = False
class _NamedTextIOWrapper(io.TextIOWrapper):
def __init__(
self, buffer: t.BinaryIO, name: str, mode: str, **kwargs: t.Any
) -> None:
super().__init__(buffer, **kwargs)
self._name = name
self._mode = mode
@property
def name(self) -> str:
return self._name
@property
def mode(self) -> str:
return self._mode
def make_input_stream(
input: t.Optional[t.Union[str, bytes, t.IO]], charset: str
) -> t.BinaryIO:
# Is already an input stream.
if hasattr(input, "read"):
rv = _find_binary_reader(t.cast(t.IO, input))
if rv is not None:
return rv
raise TypeError("Could not find binary reader for input stream.")
if input is None:
input = b""
elif isinstance(input, str):
input = input.encode(charset)
return io.BytesIO(t.cast(bytes, input))
class Result:
"""Holds the captured result of an invoked CLI script."""
def __init__(
self,
runner: "CliRunner",
stdout_bytes: bytes,
stderr_bytes: t.Optional[bytes],
return_value: t.Any,
exit_code: int,
exception: t.Optional[BaseException],
exc_info: t.Optional[
t.Tuple[t.Type[BaseException], BaseException, TracebackType]
] = None,
):
#: The runner that created the result
self.runner = runner
#: The standard output as bytes.
self.stdout_bytes = stdout_bytes
#: The standard error as bytes, or None if not available
self.stderr_bytes = stderr_bytes
#: The value returned from the invoked command.
#:
#: .. versionadded:: 8.0
self.return_value = return_value
#: The exit code as integer.
self.exit_code = exit_code
#: The exception that happened if one did.
self.exception = exception
#: The traceback
self.exc_info = exc_info
@property
def output(self) -> str:
"""The (standard) output as unicode string."""
return self.stdout
@property
def stdout(self) -> str:
"""The standard output as unicode string."""
return self.stdout_bytes.decode(self.runner.charset, "replace").replace(
"\r\n", "\n"
)
@property
def stderr(self) -> str:
"""The standard error as unicode string."""
if self.stderr_bytes is None:
raise ValueError("stderr not separately captured")
return self.stderr_bytes.decode(self.runner.charset, "replace").replace(
"\r\n", "\n"
)
def __repr__(self) -> str:
exc_str = repr(self.exception) if self.exception else "okay"
return f"<{type(self).__name__} {exc_str}>"
class CliRunner:
"""The CLI runner provides functionality to invoke a Click command line
script for unittesting purposes in a isolated environment. This only
works in single-threaded systems without any concurrency as it changes the
global interpreter state.
:param charset: the character set for the input and output data.
:param env: a dictionary with environment variables for overriding.
:param echo_stdin: if this is set to `True`, then reading from stdin writes
to stdout. This is useful for showing examples in
some circumstances. Note that regular prompts
will automatically echo the input.
:param mix_stderr: if this is set to `False`, then stdout and stderr are
preserved as independent streams. This is useful for
Unix-philosophy apps that have predictable stdout and
noisy stderr, such that each may be measured
independently
"""
def __init__(
self,
charset: str = "utf-8",
env: t.Optional[t.Mapping[str, t.Optional[str]]] = None,
echo_stdin: bool = False,
mix_stderr: bool = True,
) -> None:
self.charset = charset
self.env = env or {}
self.echo_stdin = echo_stdin
self.mix_stderr = mix_stderr
def get_default_prog_name(self, cli: "BaseCommand") -> str:
"""Given a command object it will return the default program name
for it. The default is the `name` attribute or ``"root"`` if not
set.
"""
return cli.name or "root"
def make_env(
self, overrides: t.Optional[t.Mapping[str, t.Optional[str]]] = None
) -> t.Mapping[str, t.Optional[str]]:
"""Returns the environment overrides for invoking a script."""
rv = dict(self.env)
if overrides:
rv.update(overrides)
return rv
@contextlib.contextmanager
def isolation(
self,
input: t.Optional[t.Union[str, bytes, t.IO]] = None,
env: t.Optional[t.Mapping[str, t.Optional[str]]] = None,
color: bool = False,
) -> t.Iterator[t.Tuple[io.BytesIO, t.Optional[io.BytesIO]]]:
"""A context manager that sets up the isolation for invoking of a
command line tool. This sets up stdin with the given input data
and `os.environ` with the overrides from the given dictionary.
This also rebinds some internals in Click to be mocked (like the
prompt functionality).
This is automatically done in the :meth:`invoke` method.
:param input: the input stream to put into sys.stdin.
:param env: the environment overrides as dictionary.
:param color: whether the output should contain color codes. The
application can still override this explicitly.
.. versionchanged:: 8.0
``stderr`` is opened with ``errors="backslashreplace"``
instead of the default ``"strict"``.
.. versionchanged:: 4.0
Added the ``color`` parameter.
"""
bytes_input = make_input_stream(input, self.charset)
echo_input = None
old_stdin = sys.stdin
old_stdout = sys.stdout
old_stderr = sys.stderr
old_forced_width = formatting.FORCED_WIDTH
formatting.FORCED_WIDTH = 80
env = self.make_env(env)
bytes_output = io.BytesIO()
if self.echo_stdin:
bytes_input = echo_input = t.cast(
t.BinaryIO, EchoingStdin(bytes_input, bytes_output)
)
sys.stdin = text_input = _NamedTextIOWrapper(
bytes_input, encoding=self.charset, name="<stdin>", mode="r"
)
if self.echo_stdin:
# Force unbuffered reads, otherwise TextIOWrapper reads a
# large chunk which is echoed early.
text_input._CHUNK_SIZE = 1 # type: ignore
sys.stdout = _NamedTextIOWrapper(
bytes_output, encoding=self.charset, name="<stdout>", mode="w"
)
bytes_error = None
if self.mix_stderr:
sys.stderr = sys.stdout
else:
bytes_error = io.BytesIO()
sys.stderr = _NamedTextIOWrapper(
bytes_error,
encoding=self.charset,
name="<stderr>",
mode="w",
errors="backslashreplace",
)
@_pause_echo(echo_input) # type: ignore
def visible_input(prompt: t.Optional[str] = None) -> str:
sys.stdout.write(prompt or "")
val = text_input.readline().rstrip("\r\n")
sys.stdout.write(f"{val}\n")
sys.stdout.flush()
return val
@_pause_echo(echo_input) # type: ignore
def hidden_input(prompt: t.Optional[str] = None) -> str:
sys.stdout.write(f"{prompt or ''}\n")
sys.stdout.flush()
return text_input.readline().rstrip("\r\n")
@_pause_echo(echo_input) # type: ignore
def _getchar(echo: bool) -> str:
char = sys.stdin.read(1)
if echo:
sys.stdout.write(char)
sys.stdout.flush()
return char
default_color = color
def should_strip_ansi(
stream: t.Optional[t.IO] = None, color: t.Optional[bool] = None
) -> bool:
if color is None:
return not default_color
return not color
old_visible_prompt_func = termui.visible_prompt_func
old_hidden_prompt_func = termui.hidden_prompt_func
old__getchar_func = termui._getchar
old_should_strip_ansi = utils.should_strip_ansi # type: ignore
termui.visible_prompt_func = visible_input
termui.hidden_prompt_func = hidden_input
termui._getchar = _getchar
utils.should_strip_ansi = should_strip_ansi # type: ignore
old_env = {}
try:
for key, value in env.items():
old_env[key] = os.environ.get(key)
if value is None:
try:
del os.environ[key]
except Exception:
pass
else:
os.environ[key] = value
yield (bytes_output, bytes_error)
finally:
for key, value in old_env.items():
if value is None:
try:
del os.environ[key]
except Exception:
pass
else:
os.environ[key] = value
sys.stdout = old_stdout
sys.stderr = old_stderr
sys.stdin = old_stdin
termui.visible_prompt_func = old_visible_prompt_func
termui.hidden_prompt_func = old_hidden_prompt_func
termui._getchar = old__getchar_func
utils.should_strip_ansi = old_should_strip_ansi # type: ignore
formatting.FORCED_WIDTH = old_forced_width
def invoke(
self,
cli: "BaseCommand",
args: t.Optional[t.Union[str, t.Sequence[str]]] = None,
input: t.Optional[t.Union[str, bytes, t.IO]] = None,
env: t.Optional[t.Mapping[str, t.Optional[str]]] = None,
catch_exceptions: bool = True,
color: bool = False,
**extra: t.Any,
) -> Result:
"""Invokes a command in an isolated environment. The arguments are
forwarded directly to the command line script, the `extra` keyword
arguments are passed to the :meth:`~clickpkg.Command.main` function of
the command.
This returns a :class:`Result` object.
:param cli: the command to invoke
:param args: the arguments to invoke. It may be given as an iterable
or a string. When given as string it will be interpreted
as a Unix shell command. More details at
:func:`shlex.split`.
:param input: the input data for `sys.stdin`.
:param env: the environment overrides.
:param catch_exceptions: Whether to catch any other exceptions than
``SystemExit``.
:param extra: the keyword arguments to pass to :meth:`main`.
:param color: whether the output should contain color codes. The
application can still override this explicitly.
.. versionchanged:: 8.0
The result object has the ``return_value`` attribute with
the value returned from the invoked command.
.. versionchanged:: 4.0
Added the ``color`` parameter.
.. versionchanged:: 3.0
Added the ``catch_exceptions`` parameter.
.. versionchanged:: 3.0
The result object has the ``exc_info`` attribute with the
traceback if available.
"""
exc_info = None
with self.isolation(input=input, env=env, color=color) as outstreams:
return_value = None
exception: t.Optional[BaseException] = None
exit_code = 0
if isinstance(args, str):
args = shlex.split(args)
try:
prog_name = extra.pop("prog_name")
except KeyError:
prog_name = self.get_default_prog_name(cli)
try:
return_value = cli.main(args=args or (), prog_name=prog_name, **extra)
except SystemExit as e:
exc_info = sys.exc_info()
e_code = t.cast(t.Optional[t.Union[int, t.Any]], e.code)
if e_code is None:
e_code = 0
if e_code != 0:
exception = e
if not isinstance(e_code, int):
sys.stdout.write(str(e_code))
sys.stdout.write("\n")
e_code = 1
exit_code = e_code
except Exception as e:
if not catch_exceptions:
raise
exception = e
exit_code = 1
exc_info = sys.exc_info()
finally:
sys.stdout.flush()
stdout = outstreams[0].getvalue()
if self.mix_stderr:
stderr = None
else:
stderr = outstreams[1].getvalue() # type: ignore
return Result(
runner=self,
stdout_bytes=stdout,
stderr_bytes=stderr,
return_value=return_value,
exit_code=exit_code,
exception=exception,
exc_info=exc_info, # type: ignore
)
@contextlib.contextmanager
def isolated_filesystem(
self, temp_dir: t.Optional[t.Union[str, os.PathLike]] = None
) -> t.Iterator[str]:
"""A context manager that creates a temporary directory and
changes the current working directory to it. This isolates tests
that affect the contents of the CWD to prevent them from
interfering with each other.
:param temp_dir: Create the temporary directory under this
directory. If given, the created directory is not removed
when exiting.
.. versionchanged:: 8.0
Added the ``temp_dir`` parameter.
"""
cwd = os.getcwd()
t = tempfile.mkdtemp(dir=temp_dir)
os.chdir(t)
try:
yield t
finally:
os.chdir(cwd)
if temp_dir is None:
try:
shutil.rmtree(t)
except OSError: # noqa: B014
pass

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,579 @@
import os
import sys
import typing as t
from functools import update_wrapper
from types import ModuleType
from ._compat import _default_text_stderr
from ._compat import _default_text_stdout
from ._compat import _find_binary_writer
from ._compat import auto_wrap_for_ansi
from ._compat import binary_streams
from ._compat import get_filesystem_encoding
from ._compat import open_stream
from ._compat import should_strip_ansi
from ._compat import strip_ansi
from ._compat import text_streams
from ._compat import WIN
from .globals import resolve_color_default
if t.TYPE_CHECKING:
import typing_extensions as te
F = t.TypeVar("F", bound=t.Callable[..., t.Any])
def _posixify(name: str) -> str:
return "-".join(name.split()).lower()
def safecall(func: F) -> F:
"""Wraps a function so that it swallows exceptions."""
def wrapper(*args, **kwargs): # type: ignore
try:
return func(*args, **kwargs)
except Exception:
pass
return update_wrapper(t.cast(F, wrapper), func)
def make_str(value: t.Any) -> str:
"""Converts a value into a valid string."""
if isinstance(value, bytes):
try:
return value.decode(get_filesystem_encoding())
except UnicodeError:
return value.decode("utf-8", "replace")
return str(value)
def make_default_short_help(help: str, max_length: int = 45) -> str:
"""Returns a condensed version of help string."""
# Consider only the first paragraph.
paragraph_end = help.find("\n\n")
if paragraph_end != -1:
help = help[:paragraph_end]
# Collapse newlines, tabs, and spaces.
words = help.split()
if not words:
return ""
# The first paragraph started with a "no rewrap" marker, ignore it.
if words[0] == "\b":
words = words[1:]
total_length = 0
last_index = len(words) - 1
for i, word in enumerate(words):
total_length += len(word) + (i > 0)
if total_length > max_length: # too long, truncate
break
if word[-1] == ".": # sentence end, truncate without "..."
return " ".join(words[: i + 1])
if total_length == max_length and i != last_index:
break # not at sentence end, truncate with "..."
else:
return " ".join(words) # no truncation needed
# Account for the length of the suffix.
total_length += len("...")
# remove words until the length is short enough
while i > 0:
total_length -= len(words[i]) + (i > 0)
if total_length <= max_length:
break
i -= 1
return " ".join(words[:i]) + "..."
class LazyFile:
"""A lazy file works like a regular file but it does not fully open
the file but it does perform some basic checks early to see if the
filename parameter does make sense. This is useful for safely opening
files for writing.
"""
def __init__(
self,
filename: str,
mode: str = "r",
encoding: t.Optional[str] = None,
errors: t.Optional[str] = "strict",
atomic: bool = False,
):
self.name = filename
self.mode = mode
self.encoding = encoding
self.errors = errors
self.atomic = atomic
self._f: t.Optional[t.IO]
if filename == "-":
self._f, self.should_close = open_stream(filename, mode, encoding, errors)
else:
if "r" in mode:
# Open and close the file in case we're opening it for
# reading so that we can catch at least some errors in
# some cases early.
open(filename, mode).close()
self._f = None
self.should_close = True
def __getattr__(self, name: str) -> t.Any:
return getattr(self.open(), name)
def __repr__(self) -> str:
if self._f is not None:
return repr(self._f)
return f"<unopened file '{self.name}' {self.mode}>"
def open(self) -> t.IO:
"""Opens the file if it's not yet open. This call might fail with
a :exc:`FileError`. Not handling this error will produce an error
that Click shows.
"""
if self._f is not None:
return self._f
try:
rv, self.should_close = open_stream(
self.name, self.mode, self.encoding, self.errors, atomic=self.atomic
)
except OSError as e: # noqa: E402
from .exceptions import FileError
raise FileError(self.name, hint=e.strerror) from e
self._f = rv
return rv
def close(self) -> None:
"""Closes the underlying file, no matter what."""
if self._f is not None:
self._f.close()
def close_intelligently(self) -> None:
"""This function only closes the file if it was opened by the lazy
file wrapper. For instance this will never close stdin.
"""
if self.should_close:
self.close()
def __enter__(self) -> "LazyFile":
return self
def __exit__(self, exc_type, exc_value, tb): # type: ignore
self.close_intelligently()
def __iter__(self) -> t.Iterator[t.AnyStr]:
self.open()
return iter(self._f) # type: ignore
class KeepOpenFile:
def __init__(self, file: t.IO) -> None:
self._file = file
def __getattr__(self, name: str) -> t.Any:
return getattr(self._file, name)
def __enter__(self) -> "KeepOpenFile":
return self
def __exit__(self, exc_type, exc_value, tb): # type: ignore
pass
def __repr__(self) -> str:
return repr(self._file)
def __iter__(self) -> t.Iterator[t.AnyStr]:
return iter(self._file)
def echo(
message: t.Optional[t.Any] = None,
file: t.Optional[t.IO] = None,
nl: bool = True,
err: bool = False,
color: t.Optional[bool] = None,
) -> None:
"""Print a message and newline to stdout or a file. This should be
used instead of :func:`print` because it provides better support
for different data, files, and environments.
Compared to :func:`print`, this does the following:
- Ensures that the output encoding is not misconfigured on Linux.
- Supports Unicode in the Windows console.
- Supports writing to binary outputs, and supports writing bytes
to text outputs.
- Supports colors and styles on Windows.
- Removes ANSI color and style codes if the output does not look
like an interactive terminal.
- Always flushes the output.
:param message: The string or bytes to output. Other objects are
converted to strings.
:param file: The file to write to. Defaults to ``stdout``.
:param err: Write to ``stderr`` instead of ``stdout``.
:param nl: Print a newline after the message. Enabled by default.
:param color: Force showing or hiding colors and other styles. By
default Click will remove color if the output does not look like
an interactive terminal.
.. versionchanged:: 6.0
Support Unicode output on the Windows console. Click does not
modify ``sys.stdout``, so ``sys.stdout.write()`` and ``print()``
will still not support Unicode.
.. versionchanged:: 4.0
Added the ``color`` parameter.
.. versionadded:: 3.0
Added the ``err`` parameter.
.. versionchanged:: 2.0
Support colors on Windows if colorama is installed.
"""
if file is None:
if err:
file = _default_text_stderr()
else:
file = _default_text_stdout()
# Convert non bytes/text into the native string type.
if message is not None and not isinstance(message, (str, bytes, bytearray)):
out: t.Optional[t.Union[str, bytes]] = str(message)
else:
out = message
if nl:
out = out or ""
if isinstance(out, str):
out += "\n"
else:
out += b"\n"
if not out:
file.flush()
return
# If there is a message and the value looks like bytes, we manually
# need to find the binary stream and write the message in there.
# This is done separately so that most stream types will work as you
# would expect. Eg: you can write to StringIO for other cases.
if isinstance(out, (bytes, bytearray)):
binary_file = _find_binary_writer(file)
if binary_file is not None:
file.flush()
binary_file.write(out)
binary_file.flush()
return
# ANSI style code support. For no message or bytes, nothing happens.
# When outputting to a file instead of a terminal, strip codes.
else:
color = resolve_color_default(color)
if should_strip_ansi(file, color):
out = strip_ansi(out)
elif WIN:
if auto_wrap_for_ansi is not None:
file = auto_wrap_for_ansi(file) # type: ignore
elif not color:
out = strip_ansi(out)
file.write(out) # type: ignore
file.flush()
def get_binary_stream(name: "te.Literal['stdin', 'stdout', 'stderr']") -> t.BinaryIO:
"""Returns a system stream for byte processing.
:param name: the name of the stream to open. Valid names are ``'stdin'``,
``'stdout'`` and ``'stderr'``
"""
opener = binary_streams.get(name)
if opener is None:
raise TypeError(f"Unknown standard stream '{name}'")
return opener()
def get_text_stream(
name: "te.Literal['stdin', 'stdout', 'stderr']",
encoding: t.Optional[str] = None,
errors: t.Optional[str] = "strict",
) -> t.TextIO:
"""Returns a system stream for text processing. This usually returns
a wrapped stream around a binary stream returned from
:func:`get_binary_stream` but it also can take shortcuts for already
correctly configured streams.
:param name: the name of the stream to open. Valid names are ``'stdin'``,
``'stdout'`` and ``'stderr'``
:param encoding: overrides the detected default encoding.
:param errors: overrides the default error mode.
"""
opener = text_streams.get(name)
if opener is None:
raise TypeError(f"Unknown standard stream '{name}'")
return opener(encoding, errors)
def open_file(
filename: str,
mode: str = "r",
encoding: t.Optional[str] = None,
errors: t.Optional[str] = "strict",
lazy: bool = False,
atomic: bool = False,
) -> t.IO:
"""This is similar to how the :class:`File` works but for manual
usage. Files are opened non lazy by default. This can open regular
files as well as stdin/stdout if ``'-'`` is passed.
If stdin/stdout is returned the stream is wrapped so that the context
manager will not close the stream accidentally. This makes it possible
to always use the function like this without having to worry to
accidentally close a standard stream::
with open_file(filename) as f:
...
.. versionadded:: 3.0
:param filename: the name of the file to open (or ``'-'`` for stdin/stdout).
:param mode: the mode in which to open the file.
:param encoding: the encoding to use.
:param errors: the error handling for this file.
:param lazy: can be flipped to true to open the file lazily.
:param atomic: in atomic mode writes go into a temporary file and it's
moved on close.
"""
if lazy:
return t.cast(t.IO, LazyFile(filename, mode, encoding, errors, atomic=atomic))
f, should_close = open_stream(filename, mode, encoding, errors, atomic=atomic)
if not should_close:
f = t.cast(t.IO, KeepOpenFile(f))
return f
def get_os_args() -> t.Sequence[str]:
"""Returns the argument part of ``sys.argv``, removing the first
value which is the name of the script.
.. deprecated:: 8.0
Will be removed in Click 8.1. Access ``sys.argv[1:]`` directly
instead.
"""
import warnings
warnings.warn(
"'get_os_args' is deprecated and will be removed in Click 8.1."
" Access 'sys.argv[1:]' directly instead.",
DeprecationWarning,
stacklevel=2,
)
return sys.argv[1:]
def format_filename(
filename: t.Union[str, bytes, os.PathLike], shorten: bool = False
) -> str:
"""Formats a filename for user display. The main purpose of this
function is to ensure that the filename can be displayed at all. This
will decode the filename to unicode if necessary in a way that it will
not fail. Optionally, it can shorten the filename to not include the
full path to the filename.
:param filename: formats a filename for UI display. This will also convert
the filename into unicode without failing.
:param shorten: this optionally shortens the filename to strip of the
path that leads up to it.
"""
if shorten:
filename = os.path.basename(filename)
return os.fsdecode(filename)
def get_app_dir(app_name: str, roaming: bool = True, force_posix: bool = False) -> str:
r"""Returns the config folder for the application. The default behavior
is to return whatever is most appropriate for the operating system.
To give you an idea, for an app called ``"Foo Bar"``, something like
the following folders could be returned:
Mac OS X:
``~/Library/Application Support/Foo Bar``
Mac OS X (POSIX):
``~/.foo-bar``
Unix:
``~/.config/foo-bar``
Unix (POSIX):
``~/.foo-bar``
Windows (roaming):
``C:\Users\<user>\AppData\Roaming\Foo Bar``
Windows (not roaming):
``C:\Users\<user>\AppData\Local\Foo Bar``
.. versionadded:: 2.0
:param app_name: the application name. This should be properly capitalized
and can contain whitespace.
:param roaming: controls if the folder should be roaming or not on Windows.
Has no affect otherwise.
:param force_posix: if this is set to `True` then on any POSIX system the
folder will be stored in the home folder with a leading
dot instead of the XDG config home or darwin's
application support folder.
"""
if WIN:
key = "APPDATA" if roaming else "LOCALAPPDATA"
folder = os.environ.get(key)
if folder is None:
folder = os.path.expanduser("~")
return os.path.join(folder, app_name)
if force_posix:
return os.path.join(os.path.expanduser(f"~/.{_posixify(app_name)}"))
if sys.platform == "darwin":
return os.path.join(
os.path.expanduser("~/Library/Application Support"), app_name
)
return os.path.join(
os.environ.get("XDG_CONFIG_HOME", os.path.expanduser("~/.config")),
_posixify(app_name),
)
class PacifyFlushWrapper:
"""This wrapper is used to catch and suppress BrokenPipeErrors resulting
from ``.flush()`` being called on broken pipe during the shutdown/final-GC
of the Python interpreter. Notably ``.flush()`` is always called on
``sys.stdout`` and ``sys.stderr``. So as to have minimal impact on any
other cleanup code, and the case where the underlying file is not a broken
pipe, all calls and attributes are proxied.
"""
def __init__(self, wrapped: t.IO) -> None:
self.wrapped = wrapped
def flush(self) -> None:
try:
self.wrapped.flush()
except OSError as e:
import errno
if e.errno != errno.EPIPE:
raise
def __getattr__(self, attr: str) -> t.Any:
return getattr(self.wrapped, attr)
def _detect_program_name(
path: t.Optional[str] = None, _main: ModuleType = sys.modules["__main__"]
) -> str:
"""Determine the command used to run the program, for use in help
text. If a file or entry point was executed, the file name is
returned. If ``python -m`` was used to execute a module or package,
``python -m name`` is returned.
This doesn't try to be too precise, the goal is to give a concise
name for help text. Files are only shown as their name without the
path. ``python`` is only shown for modules, and the full path to
``sys.executable`` is not shown.
:param path: The Python file being executed. Python puts this in
``sys.argv[0]``, which is used by default.
:param _main: The ``__main__`` module. This should only be passed
during internal testing.
.. versionadded:: 8.0
Based on command args detection in the Werkzeug reloader.
:meta private:
"""
if not path:
path = sys.argv[0]
# The value of __package__ indicates how Python was called. It may
# not exist if a setuptools script is installed as an egg. It may be
# set incorrectly for entry points created with pip on Windows.
if getattr(_main, "__package__", None) is None or (
os.name == "nt"
and _main.__package__ == ""
and not os.path.exists(path)
and os.path.exists(f"{path}.exe")
):
# Executed a file, like "python app.py".
return os.path.basename(path)
# Executed a module, like "python -m example".
# Rewritten by Python from "-m script" to "/path/to/script.py".
# Need to look at main module to determine how it was executed.
py_module = t.cast(str, _main.__package__)
name = os.path.splitext(os.path.basename(path))[0]
# A submodule like "example.cli".
if name != "__main__":
py_module = f"{py_module}.{name}"
return f"python -m {py_module.lstrip('.')}"
def _expand_args(
args: t.Iterable[str],
*,
user: bool = True,
env: bool = True,
glob_recursive: bool = True,
) -> t.List[str]:
"""Simulate Unix shell expansion with Python functions.
See :func:`glob.glob`, :func:`os.path.expanduser`, and
:func:`os.path.expandvars`.
This intended for use on Windows, where the shell does not do any
expansion. It may not exactly match what a Unix shell would do.
:param args: List of command line arguments to expand.
:param user: Expand user home directory.
:param env: Expand environment variables.
:param glob_recursive: ``**`` matches directories recursively.
.. versionadded:: 8.0
:meta private:
"""
from glob import glob
out = []
for arg in args:
if user:
arg = os.path.expanduser(arg)
if env:
arg = os.path.expandvars(arg)
matches = glob(arg, recursive=glob_recursive)
if not matches:
out.append(arg)
else:
out.extend(matches)
return out

View File

@ -0,0 +1,202 @@
Apache License
Version 2.0, January 2004
http://www.apache.org/licenses/
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
1. Definitions.
"License" shall mean the terms and conditions for use, reproduction,
and distribution as defined by Sections 1 through 9 of this document.
"Licensor" shall mean the copyright owner or entity authorized by
the copyright owner that is granting the License.
"Legal Entity" shall mean the union of the acting entity and all
other entities that control, are controlled by, or are under common
control with that entity. For the purposes of this definition,
"control" means (i) the power, direct or indirect, to cause the
direction or management of such entity, whether by contract or
otherwise, or (ii) ownership of fifty percent (50%) or more of the
outstanding shares, or (iii) beneficial ownership of such entity.
"You" (or "Your") shall mean an individual or Legal Entity
exercising permissions granted by this License.
"Source" form shall mean the preferred form for making modifications,
including but not limited to software source code, documentation
source, and configuration files.
"Object" form shall mean any form resulting from mechanical
transformation or translation of a Source form, including but
not limited to compiled object code, generated documentation,
and conversions to other media types.
"Work" shall mean the work of authorship, whether in Source or
Object form, made available under the License, as indicated by a
copyright notice that is included in or attached to the work
(an example is provided in the Appendix below).
"Derivative Works" shall mean any work, whether in Source or Object
form, that is based on (or derived from) the Work and for which the
editorial revisions, annotations, elaborations, or other modifications
represent, as a whole, an original work of authorship. For the purposes
of this License, Derivative Works shall not include works that remain
separable from, or merely link (or bind by name) to the interfaces of,
the Work and Derivative Works thereof.
"Contribution" shall mean any work of authorship, including
the original version of the Work and any modifications or additions
to that Work or Derivative Works thereof, that is intentionally
submitted to Licensor for inclusion in the Work by the copyright owner
or by an individual or Legal Entity authorized to submit on behalf of
the copyright owner. For the purposes of this definition, "submitted"
means any form of electronic, verbal, or written communication sent
to the Licensor or its representatives, including but not limited to
communication on electronic mailing lists, source code control systems,
and issue tracking systems that are managed by, or on behalf of, the
Licensor for the purpose of discussing and improving the Work, but
excluding communication that is conspicuously marked or otherwise
designated in writing by the copyright owner as "Not a Contribution."
"Contributor" shall mean Licensor and any individual or Legal Entity
on behalf of whom a Contribution has been received by Licensor and
subsequently incorporated within the Work.
2. Grant of Copyright License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
copyright license to reproduce, prepare Derivative Works of,
publicly display, publicly perform, sublicense, and distribute the
Work and such Derivative Works in Source or Object form.
3. Grant of Patent License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
(except as stated in this section) patent license to make, have made,
use, offer to sell, sell, import, and otherwise transfer the Work,
where such license applies only to those patent claims licensable
by such Contributor that are necessarily infringed by their
Contribution(s) alone or by combination of their Contribution(s)
with the Work to which such Contribution(s) was submitted. If You
institute patent litigation against any entity (including a
cross-claim or counterclaim in a lawsuit) alleging that the Work
or a Contribution incorporated within the Work constitutes direct
or contributory patent infringement, then any patent licenses
granted to You under this License for that Work shall terminate
as of the date such litigation is filed.
4. Redistribution. You may reproduce and distribute copies of the
Work or Derivative Works thereof in any medium, with or without
modifications, and in Source or Object form, provided that You
meet the following conditions:
(a) You must give any other recipients of the Work or
Derivative Works a copy of this License; and
(b) You must cause any modified files to carry prominent notices
stating that You changed the files; and
(c) You must retain, in the Source form of any Derivative Works
that You distribute, all copyright, patent, trademark, and
attribution notices from the Source form of the Work,
excluding those notices that do not pertain to any part of
the Derivative Works; and
(d) If the Work includes a "NOTICE" text file as part of its
distribution, then any Derivative Works that You distribute must
include a readable copy of the attribution notices contained
within such NOTICE file, excluding those notices that do not
pertain to any part of the Derivative Works, in at least one
of the following places: within a NOTICE text file distributed
as part of the Derivative Works; within the Source form or
documentation, if provided along with the Derivative Works; or,
within a display generated by the Derivative Works, if and
wherever such third-party notices normally appear. The contents
of the NOTICE file are for informational purposes only and
do not modify the License. You may add Your own attribution
notices within Derivative Works that You distribute, alongside
or as an addendum to the NOTICE text from the Work, provided
that such additional attribution notices cannot be construed
as modifying the License.
You may add Your own copyright statement to Your modifications and
may provide additional or different license terms and conditions
for use, reproduction, or distribution of Your modifications, or
for any such Derivative Works as a whole, provided Your use,
reproduction, and distribution of the Work otherwise complies with
the conditions stated in this License.
5. Submission of Contributions. Unless You explicitly state otherwise,
any Contribution intentionally submitted for inclusion in the Work
by You to the Licensor shall be under the terms and conditions of
this License, without any additional terms or conditions.
Notwithstanding the above, nothing herein shall supersede or modify
the terms of any separate license agreement you may have executed
with Licensor regarding such Contributions.
6. Trademarks. This License does not grant permission to use the trade
names, trademarks, service marks, or product names of the Licensor,
except as required for reasonable and customary use in describing the
origin of the Work and reproducing the content of the NOTICE file.
7. Disclaimer of Warranty. Unless required by applicable law or
agreed to in writing, Licensor provides the Work (and each
Contributor provides its Contributions) on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
implied, including, without limitation, any warranties or conditions
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
PARTICULAR PURPOSE. You are solely responsible for determining the
appropriateness of using or redistributing the Work and assume any
risks associated with Your exercise of permissions under this License.
8. Limitation of Liability. In no event and under no legal theory,
whether in tort (including negligence), contract, or otherwise,
unless required by applicable law (such as deliberate and grossly
negligent acts) or agreed to in writing, shall any Contributor be
liable to You for damages, including any direct, indirect, special,
incidental, or consequential damages of any character arising as a
result of this License or out of the use or inability to use the
Work (including but not limited to damages for loss of goodwill,
work stoppage, computer failure or malfunction, or any and all
other commercial damages or losses), even if such Contributor
has been advised of the possibility of such damages.
9. Accepting Warranty or Additional Liability. While redistributing
the Work or Derivative Works thereof, You may choose to offer,
and charge a fee for, acceptance of support, warranty, indemnity,
or other liability obligations and/or rights consistent with this
License. However, in accepting such obligations, You may act only
on Your own behalf and on Your sole responsibility, not on behalf
of any other Contributor, and only if You agree to indemnify,
defend, and hold each Contributor harmless for any liability
incurred by, or claims asserted against, such Contributor by reason
of your accepting any such warranty or additional liability.
END OF TERMS AND CONDITIONS
APPENDIX: How to apply the Apache License to your work.
To apply the Apache License to your work, attach the following
boilerplate notice, with the fields enclosed by brackets "[]"
replaced with your own identifying information. (Don't include
the brackets!) The text should be enclosed in the appropriate
comment syntax for the file format. We also recommend that a
file or class name and description of purpose be included on the
same "printed page" as the copyright notice for easier
identification within third-party archives.
Copyright [yyyy] [name of copyright owner]
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

View File

@ -0,0 +1,98 @@
Metadata-Version: 2.1
Name: dataclasses
Version: 0.8
Summary: A backport of the dataclasses module for Python 3.6
Home-page: https://github.com/ericvsmith/dataclasses
Author: Eric V. Smith
Author-email: eric@python.org
License: Apache
Platform: UNKNOWN
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: License :: OSI Approved :: Apache Software License
Classifier: Programming Language :: Python :: 3.6
Requires-Python: >=3.6, <3.7
.. image:: https://img.shields.io/pypi/v/dataclasses.svg
:target: https://pypi.org/project/dataclasses/
This is an implementation of PEP 557, Data Classes. It is a backport
for Python 3.6. Because dataclasses will be included in Python 3.7,
any discussion of dataclass features should occur on the python-dev
mailing list at https://mail.python.org/mailman/listinfo/python-dev.
At this point this repo should only be used for historical purposes
(it's where the original dataclasses discussions took place) and for
discussion of the actual backport to Python 3.6.
See https://www.python.org/dev/peps/pep-0557/ for the details of how
Data Classes work.
A test file can be found at
https://github.com/ericvsmith/dataclasses/blob/master/test/test_dataclasses.py,
or in the sdist file.
Installation
-------------
.. code-block::
pip install dataclasses
Example Usage
-------------
.. code-block:: python
from dataclasses import dataclass
@dataclass
class InventoryItem:
name: str
unit_price: float
quantity_on_hand: int = 0
def total_cost(self) -> float:
return self.unit_price * self.quantity_on_hand
item = InventoryItem('hammers', 10.49, 12)
print(item.total_cost())
Some additional tools can be found in dataclass_tools.py, included in
the sdist.
Compatibility
-------------
This backport assumes that dict objects retain their insertion order.
This is true in the language spec for Python 3.7 and greater. Since
this is a backport to Python 3.6, it raises an interesting question:
does that guarantee apply to 3.6? For CPython 3.6 it does. As of the
time of this writing, it's also true for all other Python
implementations that claim to be 3.6 compatible, of which there are
none. Any new 3.6 implementations are expected to have ordered dicts.
See the analysis at the end of this email:
https://mail.python.org/pipermail/python-dev/2017-December/151325.html
As of version 0.4, this code no longer works with Python 3.7. For 3.7,
use the built-in dataclasses module.
Release History
---------------
+---------+------------+-------------------------------------+
| Version | Date | Description |
+=========+============+=====================================+
| 0.8 | 2020-11-13 | Fix ClassVar in .replace() |
+---------+------------+-------------------------------------+
| 0.7 | 2019-10-20 | Require python 3.6 only |
+---------+------------+-------------------------------------+
| 0.6 | 2018-05-17 | Equivalent to Python 3.7.0rc1 |
+---------+------------+-------------------------------------+
| 0.5 | 2018-03-28 | Equivalent to Python 3.7.0b3 |
+---------+------------+-------------------------------------+

View File

@ -0,0 +1,8 @@
__pycache__/dataclasses.cpython-36.pyc,,
dataclasses-0.8.dist-info/INSTALLER,sha256=zuuue4knoyJ-UwPPXg8fezS7VCrXJQrAP7zeNuwvFQg,4
dataclasses-0.8.dist-info/LICENSE.txt,sha256=z8d0m5b2O9McPEK1xHG_dWgUBT6EfBDz6wA0F7xSPTA,11358
dataclasses-0.8.dist-info/METADATA,sha256=6kGy1NyHhp_Ef9tapnLb3rQIffoxLI5aClT4fWlUvIE,3328
dataclasses-0.8.dist-info/RECORD,,
dataclasses-0.8.dist-info/WHEEL,sha256=g4nMs7d-Xl9-xC9XovUrsDHGXt-FT0E17Yqo92DEfvY,92
dataclasses-0.8.dist-info/top_level.txt,sha256=g1h_lLdyfM-aL3-M-rWnNGxWv3ZmULp2HVvAlyZq5s0,12
dataclasses.py,sha256=TBt2du4qkxxXdZ5olWFzwHpim2ch4j7kKJi2VS_qoZw,45499

View File

@ -0,0 +1,5 @@
Wheel-Version: 1.0
Generator: bdist_wheel (0.34.2)
Root-Is-Purelib: true
Tag: py3-none-any

Some files were not shown because too many files have changed in this diff Show More