1 from datetime import date
3 from django.conf import settings
4 from librarian.functions import lang_code_3to2
5 from librarian.html import transform_abstrakt
6 from librarian.builders import EpubBuilder, MobiBuilder
7 from librarian.covers.marquise import MarquiseCover, LabelMarquiseCover
9 from slugify import slugify
14 "Książka, którą czytasz, pochodzi z <a href=\"https://wolnelektury.pl/\">Wolnych Lektur</a>. Naszą misją jest wspieranie dzieciaków w dostępie do lektur szkolnych oraz zachęcanie ich do czytania. Miło Cię poznać!",
15 "Podoba Ci się to, co robimy? Jesteśmy organizacją pożytku publicznego. Wesprzyj Wolne Lektury drobną wpłatą: <a href=\"https://wolnelektury.pl/towarzystwo/\">wolnelektury.pl/towarzystwo/</a>",
16 "Przyjaciele Wolnych Lektur otrzymują dostęp do prapremier wcześniej niż inni. Zadeklaruj stałą wpłatę i dołącz do Towarzystwa Przyjaciół Wolnych Lektur: <a href=\"https://wolnelektury.pl/towarzystwo/\">wolnelektury.pl/towarzystwo/</a>",
17 "Informacje o nowościach w naszej bibliotece w Twojej skrzynce mailowej? Nic prostszego, zapisz się do newslettera. Kliknij, by pozostawić swój adres e-mail: <a href=\"https://wolnelektury.pl/newsletter/zapisz-sie/\">wolnelektury.pl/newsletter/zapisz-sie/</a>",
18 "Przekaż 1% podatku na Wolne Lektury.<br/>\nKRS: 0000070056<br/>\nNazwa organizacji: Fundacja Nowoczesna Polska<br/>\nKażda wpłacona kwota zostanie przeznaczona na rozwój Wolnych Lektur."
21 description_add = '<p>Książkę polecają <a href="https://wolnelektury.pl">Wolne Lektury</a> — najpopularniejsza biblioteka on-line.</p>'
25 #BASE_URL = 'https://wydawca.legimi.com'
26 BASE_URL = 'https://panel.legimi.pl'
27 LOGIN_URL = BASE_URL + '/publishers/membership'
28 UPLOAD_URL = BASE_URL + '/administration/upload/start'
29 CREATE_URL = BASE_URL + '/publishers/publications/create'
30 EDIT_URL = BASE_URL + '/publishers/publications/edit/%s'
31 EDIT_FILES_URL = BASE_URL + '/publishers/publications/editfiles/%s'
32 EDIT_SALE_URL = BASE_URL + '/publishers/publications/editsale/%s'
35 'Dla dzieci i młodzieży': 94,
36 'Książki dla dzieci': 15,
37 'Literatura młodzieżowa': 24,
39 'Kryminał klasyczny': 31,
40 'Kryminał współczesny': 32,
41 'Kryminał historyczny': 30,
44 'Słowniki i leksykony': 14,
50 'Dwudziestolecie międzywojenne': 88,
59 'Fantastyka i sci-fi': 25,
61 'Science fiction': 27,
63 'Antyki i kolekcjonerstwo': 53,
64 'Astrologia i wróżbiarstwo': 54,
65 'Zdrowie i rodzina': 57,
67 'Medycyna i zdrowie': 58,
71 'Kultura i sztuka': 64,
74 'Eseje literackie': 49,
77 'Wakacje i podróże': 69,
80 'Obyczajowe i romanse': 93,
84 'Powieść przygodowa': 42,
85 'Współczesna powieść przygodowa': 44,
86 'Historyczna powieść przygodowa': 43,
87 'Powieść historyczna': 46,
88 'Powieść psychologiczna': 47,
89 'Powieść religijna': 45,
91 'Romans klasyczny': 38,
92 'Romans współczesny': 39,
93 'Literatura erotyczna': 40,
94 'Romans historyczny': 37,
96 'Sensacja, thriller, horror': 91,
102 'Literatura faktu, reportaże, biografie': 92,
103 'Literatura faktu': 16,
107 'Dokument, esej': 18,
108 'Historia literatury i krytyka literacka': 23,
109 'Literatura popularnonaukowa': 22,
111 'Społeczno-polityczne': 72,
112 'Poezja i dramat': 95,
115 'Religia i duchowość': 51,
116 'Nauka i nowe technologie': 98,
117 'Nauka i technika': 61,
119 'Nauki humanistyczne': 63,
120 'Technologia i Internet': 75,
121 'Specjalistyczne': 99,
122 'Biznes i finanse': 1,
127 'Rozwój osobisty': 7,
128 'Kariera i sukces zawodowy': 8,
129 'Psychologia, motywacja': 9,
135 def __init__(self, username, password, publisher_id):
136 self.username = username
137 self.password = password
138 self.publisher_id = publisher_id
143 if self._session is None:
144 session = requests.Session()
145 response = session.post(
148 'ValidationTrue': 'true',
149 'UserName': self.username,
150 'Password': self.password,
152 self._session = session
156 return self.session.get('https://wydawca.legimi.com/publishers/publications')
158 def upload(self, content):
159 response = self.session.post(
164 model = response.json()['model']
166 "name": model['Name'],
167 "token": model['Token'],
178 def send_book(self, book, changes=None):
179 wlbook = book.wldocument(librarian2=True, changes=changes)
182 cover = LabelMarquiseCover(meta, width=1200).output_file()
183 epub_file = EpubBuilder(cover=MarquiseCover, fundraising=fundraising).build(wlbook).get_file()
184 mobi_file = MobiBuilder(cover=MarquiseCover, fundraising=fundraising).build(wlbook).get_file()
188 "Author": ", ".join(p.readable() for p in meta.authors),
189 "Year": str(date.today().year),
191 'GenreId': str(self.get_genre(wlbook)),
193 'LanguageLocale': lang_code_3to2(meta.language),
195 'Description': self.get_description(wlbook),
198 isbn = meta.isbn_html
199 if isbn.upper().startswith('ISBN '):
202 book_data['Isbn'] = isbn
206 cover_data = self.upload(
207 (meta.url.slug + '.jpg', cover.get_file(), 'image/jpeg')
210 "Cover.Name": cover_data['name'],
211 "Cover.Token": cover_data['token'],
212 "Cover.Url": cover_data['url'],
215 epub_data = self.upload(
216 (meta.url.slug + '.epub', epub_file, 'application/epub+zip')
219 'BookEpub.Token': epub_data['token'],
220 'BookEpub.Name': epub_data['name'],
221 'SampleEpubType': 'Generation',
224 mobi_data = self.upload(
225 (meta.url.slug + '.mobi', mobi_file, 'application/x-mobipocket-ebook')
228 'BookMobi.Token': mobi_data['token'],
229 'BookMobi.Name': mobi_data['name'],
242 legimi_id = self.create_book(book_data, files_data)
244 book.legimi_id = legimi_id
245 book.save(update_fields=['legimi_id'])
247 def get_description(self, wlbook):
249 abstract = wlbook.tree.find('.//abstrakt')
250 if abstract is not None:
251 description = transform_abstrakt(abstract)
252 description += description_add
254 description += ', '.join(
255 '<a href="https://wolnelektury.pl/katalog/autor/{}/">{}</a>'.format(
256 slugify(p.readable()),
259 for p in wlbook.meta.authors
261 description += '<a href="https://wolnelektury.pl/katalog/lektura/{}/">{}</a><br>'.format(
262 wlbook.meta.url.slug,
265 if wlbook.meta.translators:
266 description += 'tłum. ' + ', '.join(p.readable() for p in wlbook.meta.translators) + '<br>'
267 description += 'Epoka: ' + ', '.join(
268 '<a href="https://wolnelektury.pl/katalog/epoka/{}/">{}</a>'.format(
272 for p in wlbook.meta.epochs
274 description += 'Rodzaj: ' + ', '.join(
275 '<a href="https://wolnelektury.pl/katalog/rodzaj/{}/">{}</a>'.format(
279 for p in wlbook.meta.kinds
281 description += 'Gatunek: ' + ', '.join(
282 '<a href="https://wolnelektury.pl/katalog/gatunek/{}/">{}</a>'.format(
286 for p in wlbook.meta.genres
289 if wlbook.meta.audience:
290 description += '<p><em>{}</em> to lektura szkolna.'.format(wlbook.meta.title)
291 if wlbook.tree.find('//pe') is not None:
292 description += '<br>Ebook <em>{title}</em> zawiera przypisy opracowane specjalnie dla uczennic i uczniów {school}.'.format(
293 title=wlbook.meta.title,
294 school='szkoły podstawowej' if wlbook.meta.audience == 'SP' else 'liceum i technikum'
296 description += '</p>'
299 def get_genre(self, wlbook):
300 if wlbook.meta.legimi and wlbook.meta.legimi in self.CATEGORIES:
301 return self.CATEGORIES[wlbook.meta.legimi]
302 for epoch in wlbook.meta.epochs:
303 if epoch in self.CATEGORIES:
304 return self.CATEGORIES[epoch]
305 return self.CATEGORIES['Lektury']
307 def create_book(self, book_data, files_data):
309 'createValidationTrue': 'true',
310 'PublisherId': self.publisher_id,#3609954
311 'IsLibraryPass': 'False',
313 'SamplesGenerationType': 'Quantity',
314 'SamplesGenerationPercent': '10',
316 'EnterToTheMarketType': 'No',
317 'EnterToTheMarketDate': '',
319 'SalesNoLimitOption': 'false',
320 'SalesNoLimitKindle': 'false',
321 'SalesInStoreEbookGrossValue': '0,00',
322 'SalesPromotion': 'False',
323 'SalesPromotionGrossValue': '0,00',
324 'SalesPromotionDatesRange.DateStart': '',
325 'SalesPromotionDatesRange.DateEnd': '',
328 for form in 'Epub', 'Mobi', 'Pdf':
330 f'Book{form}.Token': '',
331 f'Book{form}.Name': '',
332 f'Book{form}.StorageName': '',
333 f'Book{form}.Order': '',
335 f'Sample{form}Type': 'Files',
336 f'Sample{form}.Token': '',
337 f'Sample{form}.Name': '',
338 f'Sample{form}.StorageName': '',
339 f'Sample{form}.Order': '',
342 data.update(book_data)
343 data.update(files_data)
345 response = self.session.post(self.CREATE_URL, data=data)
346 m = re.search(r'/(\d+)$', response.url)
350 def edit(self, legimi_id, data):
352 'ValidationTrue': 'true',
359 self.EDIT_URL % legimi_id,
363 def edit_files(self, legimi_id, files_data):
365 'ValidationTrue': 'true',
367 'SamplesGenerationType': 'Quantity',
368 'SamplesGenerationPercent': '10',
371 for form in 'Epub', 'Mobi', 'Pdf':
373 f'Book{form}.Token': '',
374 f'Book{form}.Name': '',
375 f'Book{form}.StorageName': '',
376 f'Book{form}.Order': '',
378 f'Sample{form}.Type': 'Files',
379 f'Sample{form}.Token': '',
380 f'Sample{form}.Name': '',
381 f'Sample{form}.StorageName': '',
382 f'Sample{form}.Order': '',
385 current.update(files_data)
387 response = self.session.post(
388 self.EDIT_FILES_URL % legimi_id,
392 def edit_sale(self, book):
393 assert book.legimi_id
395 words = book.wldocument().get_statistics()['total']['words_with_fn']
397 price = settings.LEGIMI_SMALL_PRICE
398 if words > settings.LEGIMI_SMALL_WORDS:
399 price = settings.LEGIMI_BIG_PRICE
401 abo = 'true' if words > settings.LEGIMI_BIG_WORDS else 'false'
404 'ValidationTrue': 'true',
405 'Id': book.legimi_id,
406 'SalesPromotionId': "0",
407 'IsLibraryPass': "False",
408 'OriginalEnterToTheMarketType': "No",
409 'OriginalHidingDate': "",
410 'OriginalEnterToTheMarketDate': "",
411 'EnterToTheMarketType': "Yes",
412 'EnterToTheMarketDate': "",
414 'SalesNoLimitOption': abo,
415 'SalesNoLimitKindle': abo,
416 'SalesInStoreEbookGrossValue': f'{price},00',
417 'SalesPromotion': "False",
418 'SalesPromotionGrossValue': "0,00",
419 'SalesPromotionDatesRange.DateStart': "",
420 'SalesPromotionDatesRange.DateEnd': "",
424 self.EDIT_SALE_URL % book.legimi_id,
430 settings.LEGIMI_USERNAME,
431 settings.LEGIMI_PASSWORD,
432 settings.LEGIMI_PUBLISHER_ID,