-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain_datastore.py
718 lines (575 loc) · 22.2 KB
/
main_datastore.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
#!/usr/bin/python3
# -*- coding: utf-8 -*-
# Copyright 2018 Google LLC
#
# 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.
# [START gae_python37_app]
from flask import Flask, render_template, stream_with_context, request, Response, send_file
# https://flask.palletsprojects.com/en/2.1.x/patterns/fileuploads/
import datetime
from io import BytesIO
from io import StringIO
import json
# https://github.com/saffsd/langid.py
import langid # For identifying language of text
import random
from zipfile import ZipFile
import queue
import random
import threading
import time
import google.auth
from google.cloud import datastore
from google.cloud import tasks
#import cloudstorage as gcs
import os
from docx import Document
from docx.enum.style import WD_STYLE_TYPE
import adlamConversion
import ahomConversion
import phkConversion
import mendeConverter
from convertDoc2 import ConvertDocx
# Datastore
datastore_client = datastore.Client()
# If `entrypoint` is not defined in app.yaml, App Engine will look for an app
# called `app` in `main.py`.
app = Flask(__name__)
ts_client = tasks.CloudTasksClient()
_, PROJECT_ID = google.auth.default()
QUEUE_NAME = 'font-convert-queue'
REGION_ID = LOCATION_ID = 'us-central1'
QUEUE_PATH = ts_client.queue_path(PROJECT_ID, REGION_ID, QUEUE_NAME)
# Try using exporting threads to give progress report
exporting_threads = {}
# Global queue for messages
queue = queue.Queue(maxsize=100)
app.debug = True
@app.route('/')
def hello():
"""Top of the conversion application."""
who = request.url
return render_template('main.html', base=who)
# https://pythonbasics.org/flask-upload-files
@app.route('/upload/adlam')
def upload():
who = request.host_url
scriptIndex = request.args.get('scriptIndex', 0)
# For indexing a thread
taskId = random.randint(0, 7777)
return render_template('upload.html',
base=who,
lang='ff',
scriptIndex=scriptIndex,
taskId=taskId
)
# load file with explicit language and encoding
@app.route('/uploadlang')
def uploadLang():
who = request.host_url
lang = request.args.get('lang', 'und')
script_index = request.args.get('script_index', 0)
taskId = random.randint(0, 7777)
if lang == 'aho':
unicode_font_list = ['Noto Serif Ahom', 'Ahom Manuscript Unicode']
lang_name = 'Tai Ahom'
elif lang == 'phk':
unicode_font_list = ['Ramayana Unicode', 'Noto Sans Myanmar Light', 'Noto Serif Myanmar Thin', 'Noto Serif Myanmar Regular', 'Noto Serif Myanmar Thin']
lang_name = 'Tai Phake'
return render_template('upload_lang.html',
base=who,
lang=lang,
lang_name=lang_name,
taskId=taskId,
script_index=script_index,
fonts=unicode_font_list
)
# Global
msgToSend = 'First Message'
countSent = 0
def read_file_chunks(fd):
chunks = 0
while 1:
buf = fd.read(8192)
if buf:
yield buf
else:
break
chunks += 1
if app.debug:
progressFn('Download complete')
# A way to create progress functions with other information neede
# for communication
class ProgressClass():
def __init__(self, converter, thread=None):
self.converter = converter
self.thread = thread # May be available
self.status = "Nothing"
def send(self, message):
global queue
# create output
self.status = message
self.thread.setStatus(message)
queue.put(message)
# Simple output function for tracking processing
def progressFn(msg):
global msgToSend
global countSent
if app.debug:
print('PROGRESS: %s' % msg)
msgToSend = msg
countSent = 1
def summarizeDoc(doc):
# Find the fonts and languages of the paragraphs
return
def findDocFonts(doc):
fontsFound = {}
lang_codes = {}
if not doc:
return fontsFound
if doc.paragraphs:
fontsFound = getFontsInParagraphs(doc.paragraphs, fontsFound)
getLangsInParagraphs(doc.paragraphs, lang_codes)
for table in doc.tables:
rows = table.rows
for row in rows:
for cell in row.cells:
paragraphs = cell.paragraphs
fontsFount = getFontsInParagraphs(paragraphs, fontsFound)
getLangsInParagraphs(paragraphs, lang_codes)
sections = doc.sections
for section in sections:
try:
header = section.header
fontsFound = getFontsInParagraphs(header.paragraphs, fontsFound)
getLangsInParagraphs(header.paragraphs, lang_codes)
except:
pass
try:
footer = section.footer
fontsFound = getFontsInParagraphs(footer.paragraphs, fontsFound)
getLangsInParagraphs(footer.paragraphs, lang_codes)
except:
pass
return fontsFound, lang_codes
def getLangsInParagraphs(paragraphs, para_langs):
for p in paragraphs:
lang = langid.langid.classify(p.text)
lang_code = lang[0]
if lang_code in para_langs:
para_langs[lang_code] += 1
else:
para_langs[lang_code] = 1
def getFontsInParagraphs(paragraphs, fonts):
for p in paragraphs:
for r in p.runs:
font = r.font.name
if font in fonts:
fonts[font] += 1
else:
fonts[font] = 1
return fonts
#https://tedboy.github.io/flask/generated/flask.stream_with_context.html@
@app.route('/uploader/', methods = ['GET', 'POST'])
def upload_file():
convertDoc = False
lang = request.args.get('lang', 'und')
who = '/uploader/%s' % lang
print('WHO = %s' % who)
if request.method: # anything should work! == 'POST':
formData = request.form.to_dict()
if 'ConvertToUnicode' in formData:
print('ConvertToUnicode')
convertDoc = True
try:
taskId = int(formData['taskId'])
except:
taskId = 117
print('*** taskId = %d' % taskId)
try:
lang = formData['lang']
lang_code = lang
print('lang =' + formData['lang'])
except:
lang_code = 'ff'
lang = 'ff'
file = request.files['file'] # FileStorage object
print('FILE = %s' % file)
inputFileName = file.filename
print('inputFileName = %s' % inputFileName)
if not inputFileName:
return render_template('nofileselected.html', who=who)
baseName = os.path.splitext(inputFileName)[0]
outFileName = baseName + '_Unicode.docx'
# New thread for this id
this_thread = exporting_threads[taskId] = ExportingThread()
this_thread.start()
this_thread.status = 'Creating doc %s from upload' % inputFileName
doc, fileSize = createDocFromFile(file)
if not doc:
return render_template(
'error.html',
who=who,
error='%s %s' % ('Problem creating file', inputFileName))
this_thread.status = 'Doc ready to process'
fontsFound, para_langs = findDocFonts(doc)
if not convertDoc:
# Just show information.
print('PARA LANGS = %s' % para_langs)
return render_template(
'docinfo.html',
size="{:,}".format(fileSize),
filename=inputFileName,
paragraphs="{:,}".format(len(doc.paragraphs)),
sections=len(doc.sections),
tables=len(doc.tables),
fontDict=fontsFound,
para_langs=json.dumps(para_langs),
unicodeFont=formData['UnicodeFont']
)
this_thread.status = ('Paragraphs found: %d' % len(doc.paragraphs))
# Call conversions on the document.
langConverter = None
print('LANG = %s' % lang_code)
if lang_code == 'ff':
langConverter = adlamConversion.AdlamConverter()
print('ADLAM CONVERTER CREATED')
elif lang_code =='aho':
langConverter = ahomConversion.AhomConverter()
print('AHOM CONVERTER CREATED')
elif lang_code =='phk':
langConverter = phkConversion.PhakeConverter()
print('PHK CONVERTER CREATED')
elif lang_code =='men':
langConverter = medeConverter.MendeConverter()
print('MEN CONVERTER CREATED')
langConverter.detectLang = langid.langid
langConverter.ignoreLangs = ['en', 'fr'] # Not converted
langConverter.taskId = taskId
newProgressObj = ProgressClass(langConverter, this_thread)
try:
try:
scriptIndex = int(formData['scriptIndex'])
except:
print('NO SCRIPT INDEX')
scriptIndex = 0
langConverter.setScriptIndex(scriptIndex)
langConverter.setLowerMode(True)
langConverter.setSentenceMode(True)
paragraphs = doc.paragraphs
print(' %s PARAGRAPHS' % len(paragraphs))
msgToSend = '%d paragraphs in %s\n' % (len(paragraphs), inputFileName)
countSent = 0
except BaseException as err:
return render_template('error.html',
who=who,
error='Bad langConverter: %s' % err)
try:
docConverter = ConvertDocx(langConverter, documentIn=doc,
reportProgressObj=newProgressObj)
except BaseException as error:
print('Cannot create doc converter: %s' % error)
return render_template('error.html', who=who, error=error)
result = docConverter.processDocx()
target_stream = BytesIO()
result = doc.save(target_stream)
# Download resulting converted document
# Reset the pointer to the beginning.
target_stream.seek(0)
# Deal with non-ASCII in the file name
# outFileName = fixNonAsciiFilename(outFileName)
headerFileName = "attachment;filename=%s" % outFileName
# Check if there is word list info.
wordFrequencies = langConverter.getSortedWordList()
# Try to make this with a zip archive
# Create a .tsv file of the word frequencies
text_stream = StringIO()
if wordFrequencies:
text_stream.write('%s\t%s\n' % ('Word', 'Times in file'))
for item in wordFrequencies:
outline = '%s\t%s\n' % (item[0], item[1])
text_stream.write(outline)
text_stream.seek(0)
wordsFileName = baseName + "_words.tsv"
# Create an info file
info_stream = StringIO()
now = datetime.datetime.now()
info_stream.write('Source filename = %s\n' % inputFileName)
info_stream.write('Output filename = %s\n' % outFileName)
info_stream.write('Converted to Unicode at %s\n' %
now.strftime('%Y-%m-%d %H:%M:%S'))
info_stream.write('File size: {:,} bytes\n'.format(fileSize))
info_stream.write('{:,} paragraphs\n'.format(len(doc.paragraphs)))
info_stream.write(' %d sections\n' % len(doc.sections))
info_stream.write(' %d tables\n' % len(doc.tables))
info_stream.write(' fonts found = %s\n' % fontsFound)
info_stream.write(' unicodeFont = %s\n' % formData['UnicodeFont'])
info_stream.seek(0)
# The zipfile contents
zipStream = BytesIO()
with ZipFile(zipStream, 'w') as zf:
zf.writestr(outFileName, target_stream.read())
zf.writestr(wordsFileName, text_stream.read())
zf.writestr('%s_info.txt' % baseName, info_stream.read())
zipStream.seek(0)
zipName = baseName + '_Unicode.zip'
return send_file(zipStream, as_attachment=True,
download_name=zipName)
def createZipArchive(target_stream, headerFileName, baseName, wordFrequencies):
# Try zip file...
zipStream = BytesIO()
zf= ZipFile(zipStream, 'w')
try:
zf.writestr(headerFileName, target_stream.read())
except BaseException as err:
print('*** Cannot put doc into zip file %s' % (err))
return False
text_stream = StringIO()
for item in wordFrequencies:
addLine = '%s\t%s\n' % (item[0], item[1])
text_stream.write(addLine)
text_stream.seek(0)
frequenciesName = baseName + '_words.tsv'
zf.writestr(frequenciesName, text_stream.read())
return zipStream
@app.route('/testzip')
def testZip():
document = Document()
document.add_heading('Document Title', 0)
p = document.add_paragraph('A plain paragraph having some ')
p.add_run('bold').bold = True
p.add_run(' and some ')
p.add_run('italic.').italic = True
target_stream = BytesIO()
result = document.save(target_stream)
target_stream.seek(0)
headers = {
"Content-Disposition": 'test_doc.docx'
}
# Create a text file
text_stream = StringIO()
text_stream.write("text for testing")
text_stream.seek(0)
zipStream = BytesIO()
with ZipFile(zipStream, 'w') as zf:
zf.writestr('testDoc.docx', target_stream.read())
zf.writestr('textSample.txt', text_stream.read())
zipHeaders = {
"Content-Disposition": 'test_unicode.zip'
}
zipStream.seek(0)
return send_file(zipStream, as_attachment=True,
download_name='testdoc.zip')
# return Response(
# zipStream,
# # stream_with_context(read_file_chunks(zf)),
# mimetype="application/zip",
# headers=zipHeaders
# )
return
def fixNonAsciiFilename(filename):
charList = []
for i in [ord(x) for x in filename]:
if i < 128:
charList.append(chr(i))
else:
charList.append('\\u%04x' % i)
return ''.join(charList)
# get uploaded file into document form
def createDocFromFile(file):
try:
text = file.stream.read()
data = BytesIO(text)
count = len(text)
doc = Document(data)
data.close()
return doc, count
except BaseException as err:
print('Cannot create Docx for %s. Err = %s' % (file, err))
return None, -1
@app.route('/convertAdlam', methods = ['GET', 'POST'])
def convertAdlam():
if request.method == 'POST':
formData = request.form.to_dict()
file = request.files['file'] # FileStorage object
fileName = file.filename
outFileName = os.path.splitext(fileName)[0] + '_Unicode.docx'
doc, count = createDocFromFile(file)
try:
langConverter = adlamConversion.AdlamConverter()
#
langConverter.setScriptIndex(0)
langConverter.setLowerMode(True)
langConverter.setSentenceMode(True)
paragraphs = doc.paragraphs
count = len(paragraphs)
msgToSend = '%d paragraphs in %s\n' % (count, fileName)
countSent = 0
except BaseException as err:
return 'Bad Adlam converter. Err = %s' % err
try:
docConverter = ConvertDocx(langConverter, documentIn=doc,
reportProgressObj=newProgressObj)
if docConverter:
result = docConverter.processDocx()
target_stream = BytesIO()
result = doc.save(target_stream)
try:
wordFrequencies = converter.getSortedWordList()
if wordFrequencies:
# Do something with this information
words = convertwordFrequencies.keys()
for item in words:
print(word)
except:
print('FAILED TO GET WORD LIST')
words = None
# Download resulting converted document
# Reset the pointer to the beginning.
target_stream.seek(0)
# Deal with non-ASCII in the file name
outFileName = outFileName.encode(
'ascii', errors='backslashreplace')
headerFileName = "attachment;filename=%s" % outFileName
headers = {
"Content-Disposition": headerFileName
}
try:
msgToSend += 'Starting download\n'
countSent = 1
return Response(
stream_with_context(read_file_chunks(target_stream)),
mimetype="application/vnd.openxmlformats-officnedocument.wordprocessingml.document",
headers=headers
)
except BaseException as err:
print('**** Response download failure. Err = %s' % err)
except BaseeException as err:
return 'Conversion failed. with err %s' % err
# return '%s file uploaded successfully with %d paragraphs' % (file.filename, count)
# https://stackoverflow.com/questions/12232304/how-to-implement-server-push-in-flask-framework
def event_stream():
global msgToSend
global countSent
print('STREAM count = %d, msg=%s' % (countSent, msgToSend))
if countSent < 1:
print('STREAM msg:%s' % msgToSend)
yield "data: {}\n\n".format(msgToSend);
msgToSend = msgToSend + '.'
countSent += 1
# One way to push data to client. Not working right.
# @app.route('/stream')
# def stream():
# return Response(event_stream(), mimetype="text/event-stream")
# Set up a thread to allow status updates.
# https://codehunter.cc/a/flask/flask-app-update-progress-bar-while-function-runs
class ExportingThread(threading.Thread):
def __init__(self):
self.progress = 0
super().__init__()
self.status = "moving on"
def run(self):
# Your exporting stuff goes here ...
while True: # Wait for something in the queue.
message = queue.get()
self.progress += 10
#print('THREAD QUEUE MESSAGE = %s' % message)
#print('THREAD STATUS: %s' % (self.status))
def setStatus(self, newMessage):
self.status = newMessage
@app.route('/start')
def index():
global exporting_threads
thread_id = random.randint(0, 10000)
exporting_threads[thread_id] = ExportingThread()
exporting_threads[thread_id].start()
print('THREAD ID = %d' % thread_id)
return 'task id: #%s' % thread_id
@app.route('/progress/<int:thread_id>')
def progress(thread_id):
global exporting_threads
# Gets the latest status of the thread
if thread_id in exporting_threads:
return str(exporting_threads[thread_id].status)
else:
return str('Thread %s not found' % thread_id)
@app.route('/lang')
def testLangId():
args = request.args
text = args['text']
if text:
result = langid.classify(text)
return str(result)
else:
return str("No text")
@app.route('/example_task_handler', methods=['POST'])
def example_task_handler():
"""Log the request payload."""
payload = request.get_data(as_text=True) or '(empty payload)'
print('Received task with payload: {}'.format(payload))
return 'Printed task payload: {}'.format(payload)
# [END cloud_tasks_appengine_quickstart]
@app.route('/dbtest/')
def test_datastore():
# Simple function to add something to the database
# datastore_client = datastore.Client()
print('DATASTORE CLIENT: %s' % datastore_client)
kind = "langtag"
tags = ['aho', 'ff', 'ff-Latn', 'ff-Adlm', 'phk', 'men']
lang_entry = {}
try:
lang_key = datastore_client.key(kind, tags[0])
print('LANG_KEY: %s' % lang_key)
lang_entry = datastore.Entity(key=lang_key)
print('LANG_entry: %s' % lang_entry)
except BaseException as error:
print('datastore error: %s' % error)
try:
lang_entry['name'] = 'Tai Ahom'
ahomConv = ahomConversion.AhomConverter()
lang_entry['script_range'] = [ahomConv.first, ahomConv.last]
except BaseException as error:
print('Failed on lang_entry setting %s: %s' % (lang_entry, error))
try:
datastore_client.put(lang_entry)
except BaseException as error:
print('datastore put error: %s' % error)
print(f'Saved %s for %s' % (lang_key,lang_entry['name']))
@app.route('/dbget/')
def get_datastore():
# Retrieve info from datatstore.
args = request.args
lang_tag = args['lang']
print('Lang tag = %s' % lang_tag)
print('DATASTORE CLIENT: %s' % datastore_client)
kind = "langtag"
retrieved = None
try:
lang_key = datastore_client.key(kind, lang_tag)
print('LANG_KEY: %s' % lang_key)
retrieved = datastore_client.get(lang_key)
print('RETRIEVED: %s' % retrieved)
except BaseException as error:
print('Error retrieving from data store: %s' % error)
return None
if not retrieved:
print('RETRIEVED: %s' % 'NOTHING FOUND')
else:
print('RETRIEVED: %s' % list(retrieved.fetch()))
print('DATA RETRIEVED = %s, %s' % (retrieved['name'], retrieved['script_range']))
if __name__ == '__main__':
# This is used when running locally only. When deploying to Google App
# Engine, a webserver process such as Gunicorn will serve the app. This
# can be configured by adding an `entrypoint` to app.yaml.
app.run(host='127.0.0.1', port=8080, debug=True, threaded=True)
# [END gae_python37_app]