├── logo.png ├── resources ├── icon.png ├── settings.xml └── language │ ├── resource.language.en_gb │ └── strings.po │ └── resource.language.es_ar │ └── strings.po ├── setup.cfg ├── doc └── release.txt ├── .pre-commit-config.yaml ├── README.md ├── addon.xml ├── changelog.txt ├── LICENSE.txt └── service.py /logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ramiro/service.subtitles.subdivx/master/logo.png -------------------------------------------------------------------------------- /resources/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/ramiro/service.subtitles.subdivx/master/resources/icon.png -------------------------------------------------------------------------------- /setup.cfg: -------------------------------------------------------------------------------- 1 | [isort] 2 | profile = black 3 | default_section = THIRDPARTY 4 | 5 | [flake8] 6 | exclude = .git 7 | max-line-length = 110 8 | -------------------------------------------------------------------------------- /doc/release.txt: -------------------------------------------------------------------------------- 1 | How to release a new version of the addon: 2 | 3 | - Update version number in: 4 | - addon.xml 5 | - service.py 6 | 7 | - Update version number and add entries describing main changes in the new 8 | release to the changelog.txt file. 9 | -------------------------------------------------------------------------------- /.pre-commit-config.yaml: -------------------------------------------------------------------------------- 1 | repos: 2 | - repo: https://github.com/psf/black 3 | rev: 22.6.0 4 | hooks: 5 | - id: black 6 | - repo: https://github.com/PyCQA/isort 7 | rev: 5.10.1 8 | hooks: 9 | - id: isort 10 | - repo: https://github.com/pycqa/flake8 11 | rev: 4.0.1 12 | hooks: 13 | - id: flake8 14 | -------------------------------------------------------------------------------- /resources/settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | service.subtitles.subdivx 2 | ========================= 3 | 4 | Subdivx.com subtitle service plugin for Kodi/XBMC versions ranging 5 | 6 | - From XBMC Gotham (v14) up to Kodi Leia (v18): Versions 0.3.x (in maintenance mode) -- See `gotham` branch 7 | - From Kodi Matrix (v19) and later: Versions starting with 0.4.0 -- `master` branch 8 | 9 | Ported from the pre-Gotham version. All the credit to its authors. 10 | -------------------------------------------------------------------------------- /resources/language/resource.language.en_gb/strings.po: -------------------------------------------------------------------------------- 1 | # Kodi Media Center language file 2 | msgid "" 3 | msgstr "" 4 | "Project-Id-Version: XBMC Addons\n" 5 | "Report-Msgid-Bugs-To: alanwww1@xbmc.org\n" 6 | "POT-Creation-Date: YEAR-MO-DA HO:MI+ZONE\n" 7 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" 8 | "Last-Translator: Kodi Translation Team\n" 9 | "Language-Team: English (http://www.transifex.com/projects/p/xbmc-addons/language/en/)\n" 10 | "MIME-Version: 1.0\n" 11 | "Content-Type: text/plain; charset=UTF-8\n" 12 | "Content-Transfer-Encoding: 8bit\n" 13 | "Language: en\n" 14 | "Plural-Forms: nplurals=2; plural=(n != 1);\n" 15 | 16 | msgctxt "#30000" 17 | msgid "General" 18 | msgstr "" 19 | 20 | msgctxt "#30001" 21 | msgid "Show sub uploader nick instead of Language name" 22 | msgstr "" 23 | 24 | msgctxt "#30002" 25 | msgid "Subtitle listing cache in minutes (0 turns it off)" 26 | msgstr "" 27 | -------------------------------------------------------------------------------- /resources/language/resource.language.es_ar/strings.po: -------------------------------------------------------------------------------- 1 | # Kodi Media Center language file 2 | msgid "" 3 | msgstr "" 4 | "Project-Id-Version: XBMC Addons\n" 5 | "Report-Msgid-Bugs-To: alanwww1@xbmc.org\n" 6 | "POT-Creation-Date: YEAR-MO-DA HO:MI+ZONE\n" 7 | "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" 8 | "Last-Translator: Kodi Translation Team\n" 9 | "Language-Team: Spanish (Argentina) (http://www.transifex.com/projects/p/xbmc-addons/language/es_AR/)\n" 10 | "MIME-Version: 1.0\n" 11 | "Content-Type: text/plain; charset=UTF-8\n" 12 | "Content-Transfer-Encoding: 8bit\n" 13 | "Language: es_AR\n" 14 | "Plural-Forms: nplurals=2; plural=(n != 1);\n" 15 | 16 | msgctxt "#30000" 17 | msgid "General" 18 | msgstr "General" 19 | 20 | msgctxt "#30001" 21 | msgid "Show sub uploader nick instead of Language name" 22 | msgstr "Mostrar nick del autor del subtítulo en lugar del idioma" 23 | 24 | msgctxt "#30002" 25 | msgid "Subtitle listing cache in minutes (0 turns it off)" 26 | msgstr "Cache de lista de subtítulos [minutos]. 0 desactiva" 27 | -------------------------------------------------------------------------------- /addon.xml: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | 8 | 9 | 10 | 11 | 13 | 14 | Subdivx.com 15 | Busca y descarga subtítulos desde Subdivx.com 16 | Search and download subtitles from Subdivx.com 17 | es en 18 | all 19 | GNU GENERAL PUBLIC LICENSE. Version 2, June 1991 20 | https://forum.kodi.tv/showthread.php?tid=209064 21 | https://github.com/ramiro/service.subtitles.subdivx 22 | v0.4.0 23 | - Versión compatible con Kodi version 19 (Matrix) y posteriores 24 | 25 | 26 | resources/icon.png 27 | 28 | 29 | 30 | -------------------------------------------------------------------------------- /changelog.txt: -------------------------------------------------------------------------------- 1 | 0.4.0 2 | - First version to be submitted to Kodi v19 (Matrix) addons repository. Only 3 | compatible with Python 3. Thanks pedrochiuaua for the initiative and help 4 | 5 | 0.3.9 6 | - Adapt to new method for sending search query data (POST). Thanks 7 | TheArgentinian for the report and pedrochiuaua for the fix 8 | 9 | 0.3.8 10 | - Fix search after changes in subdivx.com. Thanks alvarezj81 for the report 11 | and pedrochiuaua for the fix 12 | 13 | 0.3.7 14 | - Fix search result entry regexp 15 | 16 | 0.3.6 17 | - Use direct download strategy discovred by user @tuxitos. Thanks! 18 | - Adapt to: subdivx.com now being served via HTTPS; changes in subtitles 19 | submissions list page HTML 20 | - Improve readability of descriptions with non-ASCII chars 21 | 22 | 0.3.5 23 | - Adapt to handle subdivx.com pagination changes 24 | - Add caching of web search results with new configurable TTL setting 25 | - Limit search results to 40. Avoid duplicate results if pagination stops 26 | working 27 | 28 | 0.3.3 29 | - Implement caching of web search results. Use a cache validity of one hour. 30 | 31 | 0.3.2 32 | - Be less verbose when removing old temporary dirs. 33 | 34 | 0.3.1 35 | - Attempt at fixing a timing bug that can happen when removing the temporary 36 | directory used for decompression of the downloaded file. Happens usually 37 | on Android. 38 | 39 | 0.3.0 40 | - Release changes since 0.2.5 41 | 42 | 0.2.13 43 | - Add 'news' section to addon.xml. Translate it and description to Spanish 44 | 45 | 0.2.12 46 | - Comply with new Kodi policies: Less verbose logging by default. Achieved 47 | by moving most of it to LOGDEBUG level 48 | 49 | 0.2.11 50 | - Remove disposable temporary directory after using it. 51 | 52 | 0.2.10 53 | - Attempt 1 at fixing decompression of RAR files on Kodi 18. This introduces 54 | a dependency on the vfs.libarchive binary addon for Kodi>=18. 55 | - Users need to activate and/or install 'Archive support' add-on manually. 56 | 57 | 0.2.5 58 | - Use html2text lib to scrape comments. Thanks Iván Ridao Freitas for help. 59 | - Fix handling of movie path on filesystem. Thanks Santiago Ibarra for report. 60 | - Enhance movie search performance and accuracy. Thanks Iván Ridao Freitas 61 | for report. 62 | - Fix unzip performance using Kodi API facilities. Thanks Iván Ridao Freitas. 63 | - More robust search for TV show episodes subtitles in the face of uneven 64 | metadata quality. 65 | - Better handling of .ssa and .sub+.idx files. Stop recognizing .txt files as 66 | subtitles files. 67 | 68 | 0.2.4 69 | - Logging: Fix misleading phrasing. 70 | - Internals: Factor out detection of compressed files. Initial maintainer 71 | docs. Better docstrings. 72 | 73 | 0.2.3 74 | - Logging: Use our prefix in paths debug log for searchability. 75 | - Internals: Remove unused import. 76 | 77 | 0.2.2 78 | - Logging: Log our version, do it using LOGNOTICE level. 79 | - (Version 0.2.1 skipped to avoid clashing with unofficial fork.) 80 | 81 | 0.2.0 82 | - Make intermediate page optional. Thanks Juan Redondo and Adrián Suárez. 83 | 84 | 0.1.0 85 | - Start using minor version component so we can use micro component for 86 | pre-releases. 87 | - Make sure the userdata/addon_data/s.s.s/ dir exists when handling tempdirs 88 | even when the user hasn't touched our setting. Thanks Adrián Suárez 89 | (upadrian) 90 | - More robust final download link RE-based extraction. 91 | - Handle .srt filenames with unicode chars more robustly. 92 | - Make rating actually helpful by reflecting relative download count. 93 | - Cleanup uploader comment for the subtitle. 94 | 95 | 0.0.8 96 | - Handle new page added by subdivx.com. Thanks github users @upadrian, 97 | @gboado and @felexx90 98 | - Simplify handling of temporary directory used to download & uncompress 99 | subtitle files by skipping the 'remove it recursively and re-create' stage 100 | which caused problems with mixed/unknown encodings, Unicode text passed in 101 | by Kodi or read from the FS when calling shutil.rmtree(). 102 | Instead, use Python's tempfile.mkdtemp() and don't remove tempdirs. 103 | - Toggle default value of setting 'Show uploader name in place of language' 104 | added in v0.0.5 to be ON. 105 | 106 | 0.0.7 107 | - Remove code added for 0.0.5 which isn't needed, had been added to workarond 108 | a Kodi Helix bug with handling of temp dir. 109 | - Implement manual search. Issue #2. Thanks Diego Garber. 110 | - Minor cleanups 111 | 112 | 0.0.6 113 | - When the new option to show nicks was set, the subtitle file written ended 114 | in '..srt' instead of '.es.srt'. Workaround this with a hack. 115 | - Remove flag column from search results table. 116 | - More code readability/maintainability work. 117 | - Copied workaround for xbmcvfs.exists() problem in Kodi 14 (Helix) from 118 | opensubtitles add-on, thanks to its authors. 119 | 120 | 0.0.5 121 | - Add first setting: Show uploader name in place of language 122 | Being a spanish subtitles site, there is no point in showing the spanish 123 | flag and literal in the first column so we put that spot of the screen to 124 | better use: There are some renowned uploaders in the Subdivx.community and, 125 | for lack of more structured information in Subdivx, it can be a very 126 | important piece of information when it comes to choosing from several 127 | subtitles. Show his/her nick there. 128 | Initially the value of the setting is False. We'll see if it's worth 129 | changing it to True in some future release. 130 | - Bullet-proof .rar uncompression as it's a popular file format on subdivx.com 131 | 132 | 0.0.4 133 | - More robust parsing of description, better logging. 134 | - Also match variation in CSS class name just in case. 135 | - Fix stripping of trailing '/' from param value. 136 | 137 | 0.0.3 138 | - Handle arbitrary casing in subs file extension, e.g. .SRT, .sRt, etc. 139 | - More code cleanup (PEP8, comment/docstring typos, logic of a helper function). 140 | 141 | 0.0.2 142 | - Fixed accented characters mojibake. 143 | - Removed or replaced code of dubious utility/robustness. 144 | - More idiomatic Python code. 145 | 146 | 0.0.1 147 | - Initial port to new Gotham structure. 148 | -------------------------------------------------------------------------------- /LICENSE.txt: -------------------------------------------------------------------------------- 1 | 2 | GNU GENERAL PUBLIC LICENSE 3 | Version 2, June 1991 4 | 5 | Copyright (C) 1989, 1991 Free Software Foundation, Inc. 6 | 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 7 | Everyone is permitted to copy and distribute verbatim copies 8 | of this license document, but changing it is not allowed. 9 | 10 | Preamble 11 | 12 | The licenses for most software are designed to take away your 13 | freedom to share and change it. By contrast, the GNU General Public 14 | License is intended to guarantee your freedom to share and change free 15 | software--to make sure the software is free for all its users. This 16 | General Public License applies to most of the Free Software 17 | Foundation's software and to any other program whose authors commit to 18 | using it. (Some other Free Software Foundation software is covered by 19 | the GNU Library General Public License instead.) You can apply it to 20 | your programs, too. 21 | 22 | When we speak of free software, we are referring to freedom, not 23 | price. Our General Public Licenses are designed to make sure that you 24 | have the freedom to distribute copies of free software (and charge for 25 | this service if you wish), that you receive source code or can get it 26 | if you want it, that you can change the software or use pieces of it 27 | in new free programs; and that you know you can do these things. 28 | 29 | To protect your rights, we need to make restrictions that forbid 30 | anyone to deny you these rights or to ask you to surrender the rights. 31 | These restrictions translate to certain responsibilities for you if you 32 | distribute copies of the software, or if you modify it. 33 | 34 | For example, if you distribute copies of such a program, whether 35 | gratis or for a fee, you must give the recipients all the rights that 36 | you have. You must make sure that they, too, receive or can get the 37 | source code. And you must show them these terms so they know their 38 | rights. 39 | 40 | We protect your rights with two steps: (1) copyright the software, and 41 | (2) offer you this license which gives you legal permission to copy, 42 | distribute and/or modify the software. 43 | 44 | Also, for each author's protection and ours, we want to make certain 45 | that everyone understands that there is no warranty for this free 46 | software. If the software is modified by someone else and passed on, we 47 | want its recipients to know that what they have is not the original, so 48 | that any problems introduced by others will not reflect on the original 49 | authors' reputations. 50 | 51 | Finally, any free program is threatened constantly by software 52 | patents. We wish to avoid the danger that redistributors of a free 53 | program will individually obtain patent licenses, in effect making the 54 | program proprietary. To prevent this, we have made it clear that any 55 | patent must be licensed for everyone's free use or not licensed at all. 56 | 57 | The precise terms and conditions for copying, distribution and 58 | modification follow. 59 | 60 | GNU GENERAL PUBLIC LICENSE 61 | TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION 62 | 63 | 0. This License applies to any program or other work which contains 64 | a notice placed by the copyright holder saying it may be distributed 65 | under the terms of this General Public License. The "Program", below, 66 | refers to any such program or work, and a "work based on the Program" 67 | means either the Program or any derivative work under copyright law: 68 | that is to say, a work containing the Program or a portion of it, 69 | either verbatim or with modifications and/or translated into another 70 | language. (Hereinafter, translation is included without limitation in 71 | the term "modification".) Each licensee is addressed as "you". 72 | 73 | Activities other than copying, distribution and modification are not 74 | covered by this License; they are outside its scope. The act of 75 | running the Program is not restricted, and the output from the Program 76 | is covered only if its contents constitute a work based on the 77 | Program (independent of having been made by running the Program). 78 | Whether that is true depends on what the Program does. 79 | 80 | 1. You may copy and distribute verbatim copies of the Program's 81 | source code as you receive it, in any medium, provided that you 82 | conspicuously and appropriately publish on each copy an appropriate 83 | copyright notice and disclaimer of warranty; keep intact all the 84 | notices that refer to this License and to the absence of any warranty; 85 | and give any other recipients of the Program a copy of this License 86 | along with the Program. 87 | 88 | You may charge a fee for the physical act of transferring a copy, and 89 | you may at your option offer warranty protection in exchange for a fee. 90 | 91 | 2. You may modify your copy or copies of the Program or any portion 92 | of it, thus forming a work based on the Program, and copy and 93 | distribute such modifications or work under the terms of Section 1 94 | above, provided that you also meet all of these conditions: 95 | 96 | a) You must cause the modified files to carry prominent notices 97 | stating that you changed the files and the date of any change. 98 | 99 | b) You must cause any work that you distribute or publish, that in 100 | whole or in part contains or is derived from the Program or any 101 | part thereof, to be licensed as a whole at no charge to all third 102 | parties under the terms of this License. 103 | 104 | c) If the modified program normally reads commands interactively 105 | when run, you must cause it, when started running for such 106 | interactive use in the most ordinary way, to print or display an 107 | announcement including an appropriate copyright notice and a 108 | notice that there is no warranty (or else, saying that you provide 109 | a warranty) and that users may redistribute the program under 110 | these conditions, and telling the user how to view a copy of this 111 | License. (Exception: if the Program itself is interactive but 112 | does not normally print such an announcement, your work based on 113 | the Program is not required to print an announcement.) 114 | 115 | These requirements apply to the modified work as a whole. If 116 | identifiable sections of that work are not derived from the Program, 117 | and can be reasonably considered independent and separate works in 118 | themselves, then this License, and its terms, do not apply to those 119 | sections when you distribute them as separate works. But when you 120 | distribute the same sections as part of a whole which is a work based 121 | on the Program, the distribution of the whole must be on the terms of 122 | this License, whose permissions for other licensees extend to the 123 | entire whole, and thus to each and every part regardless of who wrote it. 124 | 125 | Thus, it is not the intent of this section to claim rights or contest 126 | your rights to work written entirely by you; rather, the intent is to 127 | exercise the right to control the distribution of derivative or 128 | collective works based on the Program. 129 | 130 | In addition, mere aggregation of another work not based on the Program 131 | with the Program (or with a work based on the Program) on a volume of 132 | a storage or distribution medium does not bring the other work under 133 | the scope of this License. 134 | 135 | 3. You may copy and distribute the Program (or a work based on it, 136 | under Section 2) in object code or executable form under the terms of 137 | Sections 1 and 2 above provided that you also do one of the following: 138 | 139 | a) Accompany it with the complete corresponding machine-readable 140 | source code, which must be distributed under the terms of Sections 141 | 1 and 2 above on a medium customarily used for software interchange; or, 142 | 143 | b) Accompany it with a written offer, valid for at least three 144 | years, to give any third party, for a charge no more than your 145 | cost of physically performing source distribution, a complete 146 | machine-readable copy of the corresponding source code, to be 147 | distributed under the terms of Sections 1 and 2 above on a medium 148 | customarily used for software interchange; or, 149 | 150 | c) Accompany it with the information you received as to the offer 151 | to distribute corresponding source code. (This alternative is 152 | allowed only for noncommercial distribution and only if you 153 | received the program in object code or executable form with such 154 | an offer, in accord with Subsection b above.) 155 | 156 | The source code for a work means the preferred form of the work for 157 | making modifications to it. For an executable work, complete source 158 | code means all the source code for all modules it contains, plus any 159 | associated interface definition files, plus the scripts used to 160 | control compilation and installation of the executable. However, as a 161 | special exception, the source code distributed need not include 162 | anything that is normally distributed (in either source or binary 163 | form) with the major components (compiler, kernel, and so on) of the 164 | operating system on which the executable runs, unless that component 165 | itself accompanies the executable. 166 | 167 | If distribution of executable or object code is made by offering 168 | access to copy from a designated place, then offering equivalent 169 | access to copy the source code from the same place counts as 170 | distribution of the source code, even though third parties are not 171 | compelled to copy the source along with the object code. 172 | 173 | 4. You may not copy, modify, sublicense, or distribute the Program 174 | except as expressly provided under this License. Any attempt 175 | otherwise to copy, modify, sublicense or distribute the Program is 176 | void, and will automatically terminate your rights under this License. 177 | However, parties who have received copies, or rights, from you under 178 | this License will not have their licenses terminated so long as such 179 | parties remain in full compliance. 180 | 181 | 5. You are not required to accept this License, since you have not 182 | signed it. However, nothing else grants you permission to modify or 183 | distribute the Program or its derivative works. These actions are 184 | prohibited by law if you do not accept this License. Therefore, by 185 | modifying or distributing the Program (or any work based on the 186 | Program), you indicate your acceptance of this License to do so, and 187 | all its terms and conditions for copying, distributing or modifying 188 | the Program or works based on it. 189 | 190 | 6. Each time you redistribute the Program (or any work based on the 191 | Program), the recipient automatically receives a license from the 192 | original licensor to copy, distribute or modify the Program subject to 193 | these terms and conditions. You may not impose any further 194 | restrictions on the recipients' exercise of the rights granted herein. 195 | You are not responsible for enforcing compliance by third parties to 196 | this License. 197 | 198 | 7. If, as a consequence of a court judgment or allegation of patent 199 | infringement or for any other reason (not limited to patent issues), 200 | conditions are imposed on you (whether by court order, agreement or 201 | otherwise) that contradict the conditions of this License, they do not 202 | excuse you from the conditions of this License. If you cannot 203 | distribute so as to satisfy simultaneously your obligations under this 204 | License and any other pertinent obligations, then as a consequence you 205 | may not distribute the Program at all. For example, if a patent 206 | license would not permit royalty-free redistribution of the Program by 207 | all those who receive copies directly or indirectly through you, then 208 | the only way you could satisfy both it and this License would be to 209 | refrain entirely from distribution of the Program. 210 | 211 | If any portion of this section is held invalid or unenforceable under 212 | any particular circumstance, the balance of the section is intended to 213 | apply and the section as a whole is intended to apply in other 214 | circumstances. 215 | 216 | It is not the purpose of this section to induce you to infringe any 217 | patents or other property right claims or to contest validity of any 218 | such claims; this section has the sole purpose of protecting the 219 | integrity of the free software distribution system, which is 220 | implemented by public license practices. Many people have made 221 | generous contributions to the wide range of software distributed 222 | through that system in reliance on consistent application of that 223 | system; it is up to the author/donor to decide if he or she is willing 224 | to distribute software through any other system and a licensee cannot 225 | impose that choice. 226 | 227 | This section is intended to make thoroughly clear what is believed to 228 | be a consequence of the rest of this License. 229 | 230 | 8. If the distribution and/or use of the Program is restricted in 231 | certain countries either by patents or by copyrighted interfaces, the 232 | original copyright holder who places the Program under this License 233 | may add an explicit geographical distribution limitation excluding 234 | those countries, so that distribution is permitted only in or among 235 | countries not thus excluded. In such case, this License incorporates 236 | the limitation as if written in the body of this License. 237 | 238 | 9. The Free Software Foundation may publish revised and/or new versions 239 | of the General Public License from time to time. Such new versions will 240 | be similar in spirit to the present version, but may differ in detail to 241 | address new problems or concerns. 242 | 243 | Each version is given a distinguishing version number. If the Program 244 | specifies a version number of this License which applies to it and "any 245 | later version", you have the option of following the terms and conditions 246 | either of that version or of any later version published by the Free 247 | Software Foundation. If the Program does not specify a version number of 248 | this License, you may choose any version ever published by the Free Software 249 | Foundation. 250 | 251 | 10. If you wish to incorporate parts of the Program into other free 252 | programs whose distribution conditions are different, write to the author 253 | to ask for permission. For software which is copyrighted by the Free 254 | Software Foundation, write to the Free Software Foundation; we sometimes 255 | make exceptions for this. Our decision will be guided by the two goals 256 | of preserving the free status of all derivatives of our free software and 257 | of promoting the sharing and reuse of software generally. 258 | 259 | NO WARRANTY 260 | 261 | 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY 262 | FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN 263 | OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES 264 | PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED 265 | OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 266 | MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS 267 | TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE 268 | PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, 269 | REPAIR OR CORRECTION. 270 | 271 | 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING 272 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR 273 | REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, 274 | INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING 275 | OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED 276 | TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY 277 | YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER 278 | PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE 279 | POSSIBILITY OF SUCH DAMAGES. 280 | 281 | END OF TERMS AND CONDITIONS 282 | ------------------------------------------------------------------------- 283 | -------------------------------------------------------------------------------- /service.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | # Subdivx.com subtitles, based on a mod of Undertext subtitles 3 | # Adaptation: enric_godes@hotmail.com | Please use email address for your 4 | # comments 5 | # Port to XBMC 13 Gotham subtitles infrastructure: cramm, Mar 2014 6 | # Port to Kodi 19 Matrix/Python 3: pedrochiuaua, cramm, 2021-2022 7 | 8 | 9 | import os 10 | import os.path 11 | import re 12 | import shutil 13 | from json import loads 14 | from os.path import join as pjoin 15 | from pprint import pformat 16 | 17 | try: 18 | import StorageServer 19 | except Exception: 20 | import storageserverdummy as StorageServer 21 | 22 | import sys 23 | import tempfile 24 | import urllib.error 25 | import urllib.request 26 | from unicodedata import normalize 27 | from urllib.parse import parse_qs, quote, quote_plus, unquote, urlencode 28 | 29 | try: 30 | import xbmc 31 | except ImportError: 32 | if len(sys.argv) > 1 and sys.argv[1] == "test": 33 | import unittest # NOQA 34 | 35 | try: 36 | import mock # NOQA 37 | except ImportError: 38 | print("You need to install the mock Python library to run unit tests.\n") 39 | sys.exit(1) 40 | else: 41 | from xbmc import ( # noqa: F401 42 | LOGDEBUG, 43 | LOGINFO, 44 | LOGWARNING, 45 | LOGERROR, 46 | LOGFATAL, 47 | LOGNONE, 48 | ) 49 | import xbmcaddon 50 | import xbmcgui 51 | import xbmcplugin 52 | import xbmcvfs 53 | 54 | import html2text 55 | 56 | __addon__ = xbmcaddon.Addon() 57 | __author__ = __addon__.getAddonInfo("author") 58 | __scriptid__ = __addon__.getAddonInfo("id") 59 | __scriptname__ = __addon__.getAddonInfo("name") 60 | __version__ = "0.4.0" 61 | __language__ = __addon__.getLocalizedString 62 | 63 | __cwd__ = xbmcvfs.translatePath(__addon__.getAddonInfo("path")) 64 | __profile__ = xbmcvfs.translatePath(__addon__.getAddonInfo("profile")) 65 | 66 | 67 | MAIN_SUBDIVX_URL = "https://www.subdivx.com/" 68 | SEARCH_PAGE_URL = MAIN_SUBDIVX_URL + "index.php" 69 | QS_DICT = { 70 | "accion": "5", 71 | "masdesc": "", 72 | "oxdown": "1", 73 | } 74 | QS_KEY_QUERY = "buscar2" 75 | QS_KEY_PAGE = "pg" 76 | MAX_RESULTS_COUNT = 40 77 | 78 | INTERNAL_LINK_URL_BASE = "plugin://%s/?" 79 | SUB_EXTS = ["SRT", "SUB", "SSA"] 80 | HTTP_USER_AGENT = ( 81 | "User-Agent=Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) " 82 | "Chrome/53.0.2785.21 Safari/537.36" 83 | ) 84 | FORCED_SUB_SENTINELS = ["FORZADO", "FORCED"] 85 | 86 | PAGE_ENCODING = "utf-8" 87 | 88 | kodi_major_version = None 89 | 90 | 91 | # ============================ 92 | # Regular expression patterns 93 | # ============================ 94 | 95 | SUBTITLE_RE = re.compile( 96 | r""" 98 | .+?\d)\.gif"\s+class="detalle_calif"\s+name="detalle_calif"> 99 | .+?(?P.*?) 100 | .+?Downloads:(?P.+?) 101 | Cds: 102 | .+?Comentarios: 103 | .+?Subido\ por:\s*(?P.+?).+?""", 104 | re.IGNORECASE | re.DOTALL | re.VERBOSE | re.UNICODE | re.MULTILINE, 105 | ) 106 | # Named groups: 107 | # 'subdivx_id': ID to fetch the subs files 108 | # 'comment': Translation author comment, may contain filename 109 | # 'downloads': Downloads, used for ratings 110 | # 'uploader': Subdivx community member uploader nick 111 | 112 | DETAIL_PAGE_LINK_RE = re.compile( 113 | r'Bajar', 114 | re.IGNORECASE | re.DOTALL | re.MULTILINE | re.UNICODE, 115 | ) 116 | 117 | DOWNLOAD_LINK_RE = re.compile( 118 | r'bajar.php\?id=(?P.*?)&u=(?P[^"\']+?)', 119 | re.IGNORECASE | re.DOTALL | re.MULTILINE | re.UNICODE, 120 | ) 121 | 122 | 123 | # ========== 124 | # Functions 125 | # ========== 126 | def is_subs_file(fn): 127 | """Detect if the file has an extension we recognise as subtitle.""" 128 | ext = fn.split(".")[-1] 129 | return ext.upper() in SUB_EXTS 130 | 131 | 132 | def is_forced_subs_file(fn): 133 | """Detect if the file has some text in its filename we recognise as forced 134 | subtitle.""" 135 | target = ".".join(fn.split(".")[:-1]) if "." in fn else fn 136 | return any(s in target.upper() for s in FORCED_SUB_SENTINELS) 137 | 138 | 139 | def is_compressed_file(fname=None, contents=None): 140 | if contents is None: 141 | assert fname is not None 142 | contents = open(fname, "rb").read() 143 | assert len(contents) > 4 144 | header = contents[:4] 145 | if header == b"Rar!": 146 | compression_type = "RAR" 147 | elif header == b"PK\x03\x04": 148 | compression_type = "ZIP" 149 | else: 150 | compression_type = None 151 | return compression_type 152 | 153 | 154 | def log(msg, level=LOGDEBUG): 155 | fname = sys._getframe(1).f_code.co_name 156 | s = "SUBDIVX - %s: %s" % (fname, msg) 157 | xbmc.log(s, level=level) 158 | 159 | 160 | def get_url(url, query_data=None): 161 | if query_data is None: 162 | req = urllib.request.Request(url) 163 | log("Fetching %s" % url) 164 | else: 165 | urlencoded_query_data = urlencode(query_data) 166 | req = urllib.request.Request( 167 | url, data=urlencoded_query_data.encode(PAGE_ENCODING) 168 | ) 169 | log("Fetching %s POST data: %s" % (url, urlencoded_query_data)) 170 | req.add_header("User-Agent", HTTP_USER_AGENT) 171 | try: 172 | response = urllib.request.urlopen(req) 173 | content = response.read() 174 | except urllib.error.HTTPError as e: 175 | log("Failed to fetch %s (HTTP status: %d)" % (url, e.code), level=LOGWARNING) 176 | except urllib.error.URLError as e: 177 | log("Failed to fetch %s (URL error %s)" % (url, e.reason), level=LOGWARNING) 178 | except Exception as e: 179 | log("Failed to fetch %s (generic error %s)" % (url, e), level=LOGWARNING) 180 | else: 181 | return content 182 | return None 183 | 184 | 185 | def get_html_url(url, query_data=None): 186 | content = get_url(url, query_data=query_data) 187 | if content is None: 188 | return None 189 | # TODO: At some point subdivx.com started to declare UTF-8 encoding in the 190 | # Content-Type HTTP response header but meta HTML tag states latin1 and 191 | # actual encoding of HTML page contents seems to be inconsistent. 192 | # We might want to look at# BeatifulSoup UnicodeDammit detwingle (already 193 | # packed as a Kodi addon or python-ftfy for a more robust solution with 194 | # less risk of generating mojibake or dropping too much content 195 | return content.decode(PAGE_ENCODING, "ignore") 196 | 197 | 198 | def cleanup_subdivx_comment(comment): 199 | """Convert the subtitle comment HTML to plain text.""" 200 | parser = html2text.HTML2Text() 201 | parser.unicode_snob = True 202 | parser.ignore_emphasis = True 203 | parser.ignore_tables = True 204 | parser.ignore_links = True 205 | parser.body_width = 1000 206 | clean_text = parser.handle(comment) 207 | # Remove new lines manually 208 | clean_text = re.sub("\n", " ", clean_text) 209 | return clean_text.rstrip(" \t") 210 | 211 | 212 | def process_page(page_nr, srch_str, file_orig_path): 213 | log("Trying page %d" % page_nr) 214 | qs_dict = QS_DICT.copy() 215 | qs_dict[QS_KEY_QUERY] = srch_str 216 | if page_nr > 1: 217 | qs_dict[QS_KEY_PAGE] = str(page_nr) 218 | content = get_html_url(SEARCH_PAGE_URL, qs_dict) 219 | if content is None: 220 | return [], set() 221 | if not SUBTITLE_RE.search(content): 222 | return [], set() 223 | subs = [] 224 | descriptions = [] 225 | for counter, match in enumerate(SUBTITLE_RE.finditer(content)): 226 | groups = match.groupdict() 227 | 228 | subdivx_id = groups["subdivx_id"] 229 | 230 | dls = re.sub(r"[,.]", "", groups["downloads"]) 231 | downloads = int(dls) 232 | 233 | raw_desc = groups["comment"] 234 | descriptions.append(raw_desc) 235 | descr = cleanup_subdivx_comment(raw_desc) 236 | 237 | # If our actual video file's name appears in the description 238 | # then set sync to True because it has better chances of its 239 | # synchronization to match 240 | _, fn = os.path.split(file_orig_path) 241 | name, _ = os.path.splitext(fn) 242 | sync = re.search(re.escape(name), descr, re.I) is not None 243 | 244 | try: 245 | if not counter: 246 | log("Subtitles found for subdivx_id = %s:" % subdivx_id) 247 | log('"%s"' % descr) 248 | except Exception: 249 | pass 250 | item = { 251 | "descr": descr, 252 | "sync": sync, 253 | "subdivx_id": subdivx_id, 254 | "uploader": groups["uploader"], 255 | "downloads": downloads, 256 | "score": int(groups["calif"]), 257 | } 258 | subs.append(item) 259 | 260 | return subs, set(descriptions) 261 | 262 | 263 | def get_all_subs(searchstring, languageshort, file_orig_path): 264 | if languageshort != "es": 265 | return [] 266 | subs_list = [] 267 | page_nr = 1 268 | last_page = set() 269 | while True: 270 | page_results, current_page = process_page(page_nr, searchstring, file_orig_path) 271 | if not page_results: 272 | if page_nr == 1: 273 | log( 274 | "No subtitle link regexp match found in page contents", 275 | level=LOGFATAL, 276 | ) 277 | break 278 | if current_page == last_page: 279 | break 280 | subs_list.extend(page_results) 281 | if len(subs_list) >= MAX_RESULTS_COUNT: 282 | break 283 | page_nr += 1 284 | last_page = current_page 285 | 286 | # Put subs with sync=True at the top 287 | subs_list = sorted(subs_list, key=lambda s: s["sync"], reverse=True) 288 | return subs_list 289 | 290 | 291 | def compute_ratings(subs_list): 292 | """ 293 | Calculate the rating figures (from zero to five) in a relative fashion 294 | based on number of downloads. 295 | 296 | This is later converted by Kodi into a zero to five stars GUI. 297 | 298 | Ideally, we should be able to use a smarter number instead of just the 299 | download count of every subtitle but it seems in Subdivx the 'score' value 300 | has no reliable value and there isn't a user ranking system in place 301 | we could use to deduce the quality of a contribution. 302 | """ 303 | max_dl_count = 0 304 | for sub in subs_list: 305 | dl_cnt = sub.get("downloads", 0) 306 | if dl_cnt > max_dl_count: 307 | max_dl_count = dl_cnt 308 | for sub in subs_list: 309 | if max_dl_count: 310 | sub["rating"] = int((sub["downloads"] / float(max_dl_count)) * 5) 311 | else: 312 | sub["rating"] = 0 313 | log("subs_list = %s" % pformat(subs_list)) 314 | 315 | 316 | def append_subtitle(kodi_dir_handle, item, filename): 317 | if __addon__.getSetting("show_nick_in_place_of_lang") == "true": 318 | item_label = item["uploader"] 319 | else: 320 | item_label = "Spanish" 321 | if kodi_major_version >= 16: 322 | listitem = xbmcgui.ListItem(label=item_label, label2=item["descr"]) 323 | listitem.setArt( 324 | { 325 | "icon": str(item["rating"]), 326 | "thumb": "", 327 | } 328 | ) 329 | else: 330 | listitem = xbmcgui.ListItem( 331 | label=item_label, 332 | label2=item["descr"], 333 | iconImage=str(item["rating"]), 334 | thumbnailImage="", 335 | ) 336 | listitem.setProperty("sync", "true" if item["sync"] else "false") 337 | listitem.setProperty( 338 | "hearing_imp", "true" if item.get("hearing_imp", False) else "false" 339 | ) 340 | 341 | # Below arguments are optional, they can be used to pass any info needed in 342 | # download function. Anything after "action=download&" will be sent to 343 | # addon once user clicks listed subtitle to download 344 | url = INTERNAL_LINK_URL_BASE % __scriptid__ 345 | xbmc_url = build_xbmc_item_url(url, item, filename) 346 | # Add it to list, this can be done as many times as needed for all 347 | # subtitles found 348 | xbmcplugin.addDirectoryItem( 349 | handle=kodi_dir_handle, url=xbmc_url, listitem=listitem, isFolder=False 350 | ) 351 | 352 | 353 | def build_xbmc_item_url(url, item, filename): 354 | """Return an internal Kodi pseudo-url for the provided sub search result""" 355 | try: 356 | xbmc_url = url + urlencode( 357 | (("id", item["subdivx_id"]), ("filename", filename.encode("utf-8"))) 358 | ) 359 | except UnicodeEncodeError: 360 | # Well, go back to trying it with its original latin1 encoding 361 | try: 362 | subdivx_id = item["subdivx_id"].encode(PAGE_ENCODING) 363 | xbmc_url = url + urlencode( 364 | (("id", subdivx_id), ("filename", filename.encode("utf-8"))) 365 | ) 366 | except Exception: 367 | log("Problematic subdivx_id: %s" % subdivx_id) 368 | raise 369 | return xbmc_url 370 | 371 | 372 | def build_tvshow_searchstring(item): 373 | parts = ["%s" % item["tvshow"]] 374 | try: 375 | season = int(item["season"]) 376 | except Exception: 377 | pass 378 | else: 379 | parts.append(" S%#02d" % season) 380 | try: 381 | episode = int(item["episode"]) 382 | except Exception: 383 | pass 384 | else: 385 | parts.append("E%#02d" % episode) 386 | return "".join(parts) 387 | 388 | 389 | def action_search(kodi_dir_handle, item): 390 | """Called when subtitle search is requested from Kodi.""" 391 | log("item = %s" % pformat(item)) 392 | # Do what's needed to get the list of subtitles from service site 393 | # use item["some_property"] that was set earlier. 394 | # Once done, set xbmcgui.ListItem() below and pass it to 395 | # xbmcplugin.addDirectoryItem() 396 | file_original_path = item["file_original_path"] 397 | 398 | if item["manual_search"]: 399 | searchstring = unquote(item["manual_search_string"]) 400 | elif item["tvshow"]: 401 | searchstring = build_tvshow_searchstring(item) 402 | else: 403 | searchstring = "%s%s" % ( 404 | item["title"], 405 | " (%s)" % item["year"].strip("()") if item.get("year") else "", 406 | ) 407 | log("Search string = %s" % searchstring) 408 | 409 | cache_ttl_value = __addon__.getSetting("cache_ttl") 410 | try: 411 | cache_ttl = int(cache_ttl_value) 412 | except Exception: 413 | cache_ttl = 0 414 | if cache_ttl: 415 | cache = StorageServer.StorageServer( 416 | "service.subtitles.subdivx", cache_ttl / 60.0 417 | ) 418 | subs_list = cache.cacheFunction( 419 | get_all_subs, searchstring, "es", file_original_path 420 | ) 421 | else: 422 | subs_list = get_all_subs(searchstring, "es", file_original_path) 423 | 424 | compute_ratings(subs_list) 425 | 426 | for sub in subs_list: 427 | append_subtitle(kodi_dir_handle, sub, file_original_path) 428 | 429 | 430 | def _handle_compressed_subs(workdir, compressed_file, ext): 431 | """ 432 | Uncompress 'compressed_file' in 'workdir'. 433 | """ 434 | if ext == "rar" and kodi_major_version >= 18: 435 | src = "archive" + "://" + quote_plus(compressed_file) + "/" 436 | (cdirs, cfiles) = xbmcvfs.listdir(src) 437 | for cfile in cfiles: 438 | fsrc = "%s%s" % (src, cfile) 439 | xbmcvfs.copy(fsrc, workdir + cfile) 440 | else: 441 | xbmc.executebuiltin("Extract(%s, %s)" % (compressed_file, workdir), True) 442 | 443 | files = os.listdir(workdir) 444 | files = [f for f in files if is_subs_file(f)] 445 | found_files = [] 446 | for fname in files: 447 | found_files.append( 448 | {"forced": is_forced_subs_file(fname), "path": pjoin(workdir, fname)} 449 | ) 450 | if not found_files: 451 | log("Failed to unpack subtitles", level=LOGFATAL) 452 | return found_files 453 | 454 | 455 | def _save_subtitles(workdir, content): 456 | """ 457 | Save dowloaded file whose content is in 'content' to a temporary file 458 | If it's a compressed one then uncompress it. 459 | 460 | Returns filename of saved file or None. 461 | """ 462 | ctype = is_compressed_file(contents=content) 463 | is_compressed = ctype is not None 464 | # Never found/downloaded an unpacked subtitles file, but just to be sure ... 465 | # Assume unpacked sub file is a '.srt' 466 | cfext = {"RAR": "rar", "ZIP": "zip"}.get(ctype, "srt") 467 | tmp_fname = pjoin(workdir, "subdivx." + cfext) 468 | log("Saving subtitles to '%s'" % tmp_fname) 469 | try: 470 | with open(tmp_fname, "wb") as fh: 471 | fh.write(content) 472 | except Exception: 473 | log("Failed to save subtitles to '%s'" % tmp_fname, level=LOGFATAL) 474 | return [] 475 | else: 476 | if is_compressed: 477 | return _handle_compressed_subs(workdir, tmp_fname, cfext) 478 | return [{"path": tmp_fname, "forced": False}] 479 | 480 | 481 | def method_traditional(sub_id, u): 482 | actual_subtitle_file_url = MAIN_SUBDIVX_URL + "bajar.php?id=" + sub_id + "&u=" + u 483 | return get_url(actual_subtitle_file_url) 484 | 485 | 486 | def method_direct_download(sub_id, u): 487 | if u == "1": 488 | u = "" 489 | for ext in (".rar", ".zip"): 490 | actual_subtitle_file_url = MAIN_SUBDIVX_URL + "sub" + u + "/" + sub_id + ext 491 | content = get_url(actual_subtitle_file_url) 492 | if content is not None: 493 | break 494 | else: 495 | return None 496 | return content 497 | 498 | 499 | def action_download(subdivx_id, workdir): 500 | """Called when subtitle download is requested from Kodi.""" 501 | # Get the page with the subtitle link, 502 | # i.e. http://www.subdivx.com/X6XMjE2NDM1X-iron-man-2-2010 503 | subtitle_detail_url = MAIN_SUBDIVX_URL + quote(subdivx_id) 504 | # Fetch and scrape [new] intermediate page 505 | html_content = get_html_url(subtitle_detail_url) 506 | if html_content is None: 507 | log( 508 | "No content found in selected subtitle intermediate detail/final download page", 509 | level=LOGFATAL, 510 | ) 511 | return [] 512 | match = DETAIL_PAGE_LINK_RE.search(html_content) 513 | if match is None: 514 | log( 515 | "Intermediate detail page for selected subtitle or expected content not " 516 | "found. Handling it as final download page" 517 | ) 518 | else: 519 | id_ = match.group("id") 520 | # Fetch and scrape final page 521 | html_content = get_html_url(MAIN_SUBDIVX_URL + id_) 522 | if html_content is None: 523 | log("No content found in final download page", level=LOGFATAL) 524 | return [] 525 | match = DOWNLOAD_LINK_RE.search(html_content) 526 | if match is None: 527 | log("Expected content not found in final download page", level=LOGFATAL) 528 | return [] 529 | id_, u = match.group("id", "u") 530 | methods = [ 531 | method_direct_download, 532 | method_traditional, 533 | ] 534 | for method in methods: 535 | content = method(id_, u) 536 | if content is not None: 537 | saved_fnames = _save_subtitles(workdir, content) 538 | break 539 | else: 540 | log("Got no content when downloading actual subtitle file", level=LOGFATAL) 541 | return [] 542 | return saved_fnames 543 | 544 | 545 | def _double_dot_fix_hack(video_filename): 546 | """Corrects filename of downloaded subtitle from Foo-Blah..srt to Foo-Blah.es.srt""" 547 | 548 | log("video_filename = %s" % video_filename) 549 | 550 | work_path = video_filename 551 | if _subtitles_setting("storagemode"): 552 | custom_subs_path = _subtitles_setting("custompath") 553 | if custom_subs_path: 554 | _, fname = os.path.split(video_filename) 555 | work_path = pjoin(custom_subs_path, fname) 556 | 557 | log("work_path = %s" % work_path) 558 | parts = work_path.rsplit(".", 1) 559 | if len(parts) > 1: 560 | rest = parts[0] 561 | for ext in ("srt", "ssa", "sub", "idx"): 562 | bad = rest + ".." + ext 563 | old = rest + ".es." + ext 564 | if xbmcvfs.exists(bad): 565 | log("%s exists" % bad) 566 | if xbmcvfs.exists(old): 567 | log("%s exists, removing" % old) 568 | xbmcvfs.delete(old) 569 | log("renaming %s to %s" % (bad, old)) 570 | xbmcvfs.rename(bad, old) 571 | 572 | 573 | def _subtitles_setting(name): 574 | """ 575 | Uses Kodi JSON-RPC API to retrieve subtitles location settings values. 576 | """ 577 | command = """{ 578 | "jsonrpc": "2.0", 579 | "id": 1, 580 | "method": "Settings.GetSettingValue", 581 | "params": { 582 | "setting": "subtitles.%s" 583 | } 584 | }""" 585 | result = xbmc.executeJSONRPC(command % name) 586 | py = loads(result) 587 | if "result" in py and "value" in py["result"]: 588 | return py["result"]["value"] 589 | else: 590 | raise ValueError 591 | 592 | 593 | def get_params(argv): 594 | params = {} 595 | qs = argv[2].lstrip("?") 596 | if qs: 597 | if qs.endswith("/"): 598 | qs = qs[:-1] 599 | parsed = parse_qs(qs) 600 | for k, v in parsed.items(): 601 | params[k] = v[0] 602 | return params 603 | 604 | 605 | def debug_dump_path(victim, name): 606 | t = type(victim) 607 | xbmc.log("SUBDIVX - %s (%s): %s" % (name, t, victim)) 608 | 609 | 610 | def _cleanup_tempdir(dir_path, verbose=False): 611 | try: 612 | shutil.rmtree(dir_path, ignore_errors=True) 613 | except Exception: 614 | if verbose: 615 | log("Failed to remove %s" % dir_path, level=LOGWARNING) 616 | return False 617 | return True 618 | 619 | 620 | def _cleanup_tempdirs(profile_path): 621 | dirs, _ = xbmcvfs.listdir(profile_path) 622 | total, ok = 0, 0 623 | for total, dir_path in enumerate(dirs[:10]): 624 | result = _cleanup_tempdir(os.path.join(profile_path, dir_path), verbose=False) 625 | if result: 626 | ok += 1 627 | log("Results: %d of %d dirs removed" % (ok, total + 1)) 628 | 629 | 630 | def sleep(secs): 631 | """Sleeps efficiently for secs seconds""" 632 | if kodi_major_version > 13: 633 | xbmc.Monitor().waitForAbort(secs) 634 | else: 635 | xbmc.sleep(1000 * secs) 636 | 637 | 638 | def main(): 639 | """Main entry point of the script when it is invoked by Kodi.""" 640 | global kodi_major_version 641 | # Get parameters from Kodi and launch actions 642 | kodi_dir_handle = int(sys.argv[1]) 643 | params = get_params(sys.argv) 644 | action = params.get("action", "Unknown") 645 | xbmc.log( 646 | "SUBDIVX - Version: %s -- Action: %s" % (__version__, action), level=LOGINFO 647 | ) 648 | kodi_major_version = int(xbmc.getInfoLabel("System.BuildVersion").split(".")[0]) 649 | 650 | if action in ("search", "manualsearch"): 651 | item = { 652 | "temp": False, 653 | "rar": False, 654 | "year": xbmc.getInfoLabel("VideoPlayer.Year"), 655 | "season": xbmc.getInfoLabel("VideoPlayer.Season"), 656 | "episode": xbmc.getInfoLabel("VideoPlayer.Episode"), 657 | "tvshow": normalize("NFKD", xbmc.getInfoLabel("VideoPlayer.TVshowtitle")), 658 | # Try to get original title 659 | "title": normalize("NFKD", xbmc.getInfoLabel("VideoPlayer.OriginalTitle")), 660 | # Full path of a playing file 661 | "file_original_path": unquote(xbmc.Player().getPlayingFile()), 662 | "3let_language": [], 663 | "2let_language": [], 664 | "manual_search": "searchstring" in params, 665 | } 666 | 667 | if "searchstring" in params: 668 | item["manual_search_string"] = params["searchstring"] 669 | 670 | for lang in unquote(params["languages"]).split(","): 671 | item["3let_language"].append(xbmc.convertLanguage(lang, xbmc.ISO_639_2)) 672 | item["2let_language"].append(xbmc.convertLanguage(lang, xbmc.ISO_639_1)) 673 | 674 | if not item["title"]: 675 | # No original title, get just Title 676 | item["title"] = normalize("NFKD", xbmc.getInfoLabel("VideoPlayer.Title")) 677 | 678 | if "s" in item["episode"].lower(): 679 | # Check if season is "Special" 680 | item["season"] = "0" 681 | item["episode"] = item["episode"][-1:] 682 | 683 | if "http" in item["file_original_path"]: 684 | item["temp"] = True 685 | 686 | elif "rar://" in item["file_original_path"]: 687 | item["rar"] = True 688 | item["file_original_path"] = os.path.dirname(item["file_original_path"][6:]) 689 | 690 | elif "stack://" in item["file_original_path"]: 691 | stackPath = item["file_original_path"].split(" , ") 692 | item["file_original_path"] = stackPath[0][8:] 693 | 694 | action_search(kodi_dir_handle, item) 695 | # Send end of directory to Kodi 696 | xbmcplugin.endOfDirectory(kodi_dir_handle) 697 | 698 | elif action == "download": 699 | debug_dump_path( 700 | xbmcvfs.translatePath(__addon__.getAddonInfo("profile")), 701 | "xbmcvfs.translatePath(__addon__.getAddonInfo('profile'))", 702 | ) 703 | debug_dump_path(__profile__, "__profile__") 704 | xbmcvfs.mkdirs(__profile__) 705 | _cleanup_tempdirs(__profile__) 706 | workdir = tempfile.mkdtemp(dir=__profile__) 707 | # Make sure it ends with a path separator (Kodi 14) 708 | workdir = workdir + os.path.sep 709 | debug_dump_path(workdir, "workdir") 710 | # We pickup our arguments sent from the action_search() function 711 | subs = action_download(params["id"], workdir) 712 | # We can return more than one subtitle for multi CD versions, for now 713 | # we are still working out how to handle that in Kodi core 714 | for sub in subs: 715 | # XXX: Kodi still can't handle multiple subtitles files returned 716 | # from an addon, it will always use the first file returned. So 717 | # there is no point in reporting a forced subtitle file to it. 718 | # See https://github.com/ramiro/service.subtitles.subdivx/issues/14 719 | if sub["forced"]: 720 | continue 721 | listitem = xbmcgui.ListItem(label=sub["path"]) 722 | xbmcplugin.addDirectoryItem( 723 | handle=kodi_dir_handle, 724 | url=sub["path"], 725 | listitem=listitem, 726 | isFolder=False, 727 | ) 728 | # Send end of directory to Kodi 729 | xbmcplugin.endOfDirectory(kodi_dir_handle) 730 | 731 | sleep(2) 732 | if __addon__.getSetting("show_nick_in_place_of_lang") == "true": 733 | _double_dot_fix_hack(params["filename"]) 734 | _cleanup_tempdir(workdir, verbose=True) 735 | 736 | 737 | if __name__ == "__main__": 738 | main() 739 | --------------------------------------------------------------------------------