Skip to content

Commit 00cf349

Browse files
Merge pull request #212 from chinapandaman/PPF-211
PPF-211: tests/docs for fill sejda
2 parents 9f40a77 + 12f69ff commit 00cf349

14 files changed

+408
-18
lines changed

PyPDFForm/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,4 @@
44

55
PyPDFForm = Wrapper
66

7-
__version__ = "0.3.6"
7+
__version__ = "0.3.7"

PyPDFForm/core/template.py

+17-7
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,21 @@
1515
class Template:
1616
"""Contains methods for interacting with a pdfrw parsed PDF form."""
1717

18+
@staticmethod
19+
def remove_all_elements(
20+
pdf: bytes
21+
) -> bytes:
22+
"""Removes all elements from a pdfrw parsed PDF form."""
23+
24+
pdf = pdfrw.PdfReader(fdata=pdf)
25+
26+
for i in range(len(pdf.pages)):
27+
elements = pdf.pages[i][TemplateCoreConstants().annotation_key]
28+
for j in reversed(range(len(elements))):
29+
elements.pop(j)
30+
31+
return Utils().generate_stream(pdf)
32+
1833
@staticmethod
1934
def iterate_elements(
2035
pdf: Union[bytes, "pdfrw.PdfReader"], sejda: bool = False
@@ -188,7 +203,7 @@ def get_draw_text_coordinates(
188203
- 2,
189204
)
190205

191-
def assign_uuid(self, pdf: bytes, sejda: bool = False) -> bytes:
206+
def assign_uuid(self, pdf: bytes) -> bytes:
192207
"""Appends a separator and uuid after each element's annotated name."""
193208

194209
_uuid = uuid.uuid4().hex
@@ -208,11 +223,6 @@ def assign_uuid(self, pdf: bytes, sejda: bool = False) -> bytes:
208223
base_key, MergeConstants().separator, existed_uuid or _uuid
209224
)
210225
}
211-
if sejda:
212-
element[TemplateCoreConstants().parent_key].update(
213-
pdfrw.PdfDict(**update_dict)
214-
)
215-
else:
216-
element.update(pdfrw.PdfDict(**update_dict))
226+
element.update(pdfrw.PdfDict(**update_dict))
217227

218228
return Utils().generate_stream(pdf_file)

PyPDFForm/middleware/wrapper.py

+8-6
Original file line numberDiff line numberDiff line change
@@ -52,9 +52,7 @@ def __init__(
5252
self.stream = template
5353
self.simple_mode = simple_mode
5454
self.sejda = sejda
55-
self.fill = self._simple_fill if simple_mode else self._fill
56-
if sejda:
57-
self.fill = self._fill
55+
self.fill = self._simple_fill if simple_mode and not sejda else self._fill
5856

5957
if not simple_mode or sejda:
6058
self.elements = {}
@@ -89,8 +87,8 @@ def __add__(self, other: "PyPDFForm") -> "PyPDFForm":
8987
TemplateMiddleware().validate_stream(self.stream)
9088
TemplateMiddleware().validate_stream(other.stream)
9189

92-
pdf_one = TemplateCore().assign_uuid(self.stream, self.sejda)
93-
pdf_two = TemplateCore().assign_uuid(other.stream, self.sejda)
90+
pdf_one = TemplateCore().assign_uuid(self.stream) if not self.sejda else self.stream
91+
pdf_two = TemplateCore().assign_uuid(other.stream) if not other.sejda else other.stream
9492

9593
new_obj = self.__class__()
9694
new_obj.stream = UtilsCore().merge_two_pdfs(pdf_one, pdf_two)
@@ -121,7 +119,11 @@ def _fill(
121119
self.elements[key].validate_value()
122120
self.elements[key].validate_text_attributes()
123121

124-
self.stream = FillerCore().fill(self.stream, self.elements, self.sejda)
122+
_fill_result = FillerCore().fill(self.stream, self.elements, self.sejda)
123+
if self.sejda:
124+
_fill_result = TemplateCore().remove_all_elements(_fill_result)
125+
126+
self.stream = _fill_result
125127

126128
return self
127129

docs/v2/api_reference.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ mode on and off yields different interactions with the fill method.
1919

2020
* **simple_mode** - a simple mode PyPDFForm object only allows filling data without specifying
2121
details like font size. Turning simple mode on also allows leaving PDF editable
22-
after filling.
22+
after filling. NOTE: `simple_mode` is not available when `sejda` is set to `True`.
2323

2424
* **global_font** - a string which sets the global font for text filled on the PDF form. The
2525
font set by this parameter has to be registered first. This will only take effect if `simple_mode` is `False`.
@@ -39,6 +39,9 @@ filled on the PDF form. This will only take effect if `simple_mode` is `False`.
3939
* **global_text_wrap_length** - an integer value which sets the global maximum number of characters before
4040
wrapping to a new line for texts
4141
filled on the PDF form. This will only take effect if `simple_mode` is `False`.
42+
43+
* **sejda** - This boolean parameter should be set to `True` if the PDF form template is prepared using Sejda.
44+
NOTE: enabling this will disable `simple_mode` even if it's set to `True`.
4245

4346
### *PyPDFForm()* **+** *PyPDFForm()*
4447

@@ -120,7 +123,7 @@ Its values currently support the following:
120123
NOTE: Only groups of radio buttons with the same name are supported. If there is only one
121124
radio button with a name, please consider using `checkbox` instead.
122125

123-
* **editable** - only available if `simple_mode` is `True`, enabling this will allow the filled PDF to be still
126+
* **editable** - only available if `simple_mode` is `True` and `sejda` is `False`, enabling this will allow the filled PDF to be still
124127
editable.
125128

126129
### **read**()

docs/v2/examples.md

+63-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,10 @@ can be used out of box.
77

88
The most common tool to create a PDF form is Adobe Acrobat. A tutorial can be found
99
[here](https://helpx.adobe.com/acrobat/using/creating-distributing-pdf-forms.html).
10-
There are other free alternatives like [this](https://pdf.wondershare.com/) that support the same functionalities.
10+
There are other free alternatives like [sejda](https://www.sejda.com/) that support similar functionalities.
11+
12+
NOTE: Sejda is highly recommended as PyPDFForm
13+
provides more stable support to PDF forms prepared using it with a `sejda` mode.
1114

1215
Unless otherwise specified, all examples will use the same PDF form which can be
1316
found [here](https://github.com/chinapandaman/PyPDFForm/blob/master/pdf_samples/v2/sample_template.pdf).
@@ -440,3 +443,62 @@ with open(PATH_TO_FILLED_PDF_FORM, "wb+") as output:
440443
```
441444

442445
Link to this example: https://github.com/chinapandaman/PyPDFForm/blob/master/examples/simple_fill_radio.py
446+
447+
448+
## Fill a PDF form prepared using Sejda
449+
450+
This example uses this [template](https://github.com/chinapandaman/PyPDFForm/blob/master/pdf_samples/v2/sample_template_sejda.pdf).
451+
It demos filling a PDF form prepared using Sejda.
452+
453+
```python
454+
import os
455+
456+
from PyPDFForm import PyPDFForm
457+
458+
PATH_TO_DOWNLOADED_SAMPLE_PDF_FORM = os.path.join(
459+
os.path.expanduser("~/Downloads"), "sample_template_sejda.pdf"
460+
) # Change this to where you downloaded the sample PDF form
461+
462+
PATH_TO_FILLED_PDF_FORM = os.path.join(
463+
os.path.expanduser("~"), "output.pdf"
464+
) # Change this to where you wish to put your filled PDF form
465+
466+
with open(PATH_TO_FILLED_PDF_FORM, "wb+") as output:
467+
output.write(
468+
PyPDFForm(PATH_TO_DOWNLOADED_SAMPLE_PDF_FORM, sejda=True)
469+
.fill(
470+
{
471+
"date": "01-01",
472+
"year": "21",
473+
"buyer_name": "John Doe",
474+
"buyer_address": "1 N Main St, Chicago, IL 60000",
475+
"seller_name": "Jack Smith",
476+
"seller_address": "2 S Main St, Chicago, IL 60000",
477+
"make": "AK",
478+
"model": "47",
479+
"caliber": "7.62-x39mm",
480+
"serial_number": "111111",
481+
"purchase_option": 0,
482+
"date_of_this_bill": True,
483+
"at_future_date": True,
484+
"other": True,
485+
"other_reason": "NO REASONS",
486+
"payment_amount": "400",
487+
"future_date": "01-01",
488+
"future_year": "22",
489+
"exchange_for": "Food",
490+
"buyer_name_printed": "John Doe",
491+
"seller_name_printed": "Jack Smith",
492+
"buyer_signed_date": "2021-01-01",
493+
"seller_signed_date": "2021-01-01",
494+
"buyer_dl_number": "D000-4609-0001",
495+
"seller_dl_number": "S530-4209-0001",
496+
"buyer_dl_state": "IL",
497+
"seller_dl_state": "IL",
498+
},
499+
)
500+
.read()
501+
)
502+
```
503+
504+
Link to this example: https://github.com/chinapandaman/PyPDFForm/blob/master/examples/fill_sejda.py

examples/fill_sejda.py

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
import os
2+
3+
from PyPDFForm import PyPDFForm
4+
5+
PATH_TO_DOWNLOADED_SAMPLE_PDF_FORM = os.path.join(
6+
os.path.expanduser("~/Downloads"), "sample_template_sejda.pdf"
7+
) # Change this to where you downloaded the sample PDF form
8+
9+
PATH_TO_FILLED_PDF_FORM = os.path.join(
10+
os.path.expanduser("~"), "output.pdf"
11+
) # Change this to where you wish to put your filled PDF form
12+
13+
with open(PATH_TO_FILLED_PDF_FORM, "wb+") as output:
14+
output.write(
15+
PyPDFForm(PATH_TO_DOWNLOADED_SAMPLE_PDF_FORM, sejda=True)
16+
.fill(
17+
{
18+
"date": "01-01",
19+
"year": "21",
20+
"buyer_name": "John Doe",
21+
"buyer_address": "1 N Main St, Chicago, IL 60000",
22+
"seller_name": "Jack Smith",
23+
"seller_address": "2 S Main St, Chicago, IL 60000",
24+
"make": "AK",
25+
"model": "47",
26+
"caliber": "7.62-x39mm",
27+
"serial_number": "111111",
28+
"purchase_option": 0,
29+
"date_of_this_bill": True,
30+
"at_future_date": True,
31+
"other": True,
32+
"other_reason": "NO REASONS",
33+
"payment_amount": "400",
34+
"future_date": "01-01",
35+
"future_year": "22",
36+
"exchange_for": "Food",
37+
"buyer_name_printed": "John Doe",
38+
"seller_name_printed": "Jack Smith",
39+
"buyer_signed_date": "2021-01-01",
40+
"seller_signed_date": "2021-01-01",
41+
"buyer_dl_number": "D000-4609-0001",
42+
"seller_dl_number": "S530-4209-0001",
43+
"buyer_dl_state": "IL",
44+
"seller_dl_state": "IL",
45+
},
46+
)
47+
.read()
48+
)
242 KB
Binary file not shown.
179 KB
Binary file not shown.

tests/conftest.py

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
# -*- coding: utf-8 -*-
2+
3+
import os
4+
5+
import pytest
6+
7+
8+
@pytest.fixture
9+
def pdf_samples():
10+
return os.path.join(os.path.dirname(__file__), "..", "pdf_samples", "v2")
11+
12+
13+
@pytest.fixture
14+
def sejda_template(pdf_samples):
15+
with open(
16+
os.path.join(pdf_samples, "sample_template_sejda.pdf"), "rb+"
17+
) as f:
18+
return f.read()
19+
20+
21+
@pytest.fixture
22+
def sejda_data():
23+
return {
24+
"date": "01-01",
25+
"year": "21",
26+
"buyer_name": "John Doe",
27+
"buyer_address": "1 N Main St, Chicago, IL 60000",
28+
"seller_name": "Jack Smith",
29+
"seller_address": "2 S Main St, Chicago, IL 60000",
30+
"make": "AK",
31+
"model": "47",
32+
"caliber": "7.62-x39mm",
33+
"serial_number": "111111",
34+
"purchase_option": 0,
35+
"date_of_this_bill": True,
36+
"at_future_date": True,
37+
"other": True,
38+
"other_reason": "NO REASONS",
39+
"payment_amount": "400",
40+
"future_date": "01-01",
41+
"future_year": "22",
42+
"exchange_for": "Food",
43+
"buyer_name_printed": "John Doe",
44+
"seller_name_printed": "Jack Smith",
45+
"buyer_signed_date": "2021-01-01",
46+
"seller_signed_date": "2021-01-01",
47+
"buyer_dl_number": "D000-4609-0001",
48+
"seller_dl_number": "S530-4209-0001",
49+
"buyer_dl_state": "IL",
50+
"seller_dl_state": "IL",
51+
}

tests/functional/test_fill.py

+13
Original file line numberDiff line numberDiff line change
@@ -308,3 +308,16 @@ def test_fill_radiobutton(pdf_samples, template_with_radiobutton_stream):
308308
)
309309

310310
assert obj.stream == f.read()
311+
312+
313+
def test_fill_sejda_and_read(sejda_template, pdf_samples, sejda_data):
314+
with open(os.path.join(pdf_samples, "sample_filled_sejda.pdf"), "rb+") as f:
315+
obj = PyPDFForm(sejda_template, sejda=True).fill(
316+
sejda_data,
317+
)
318+
assert obj.read() == obj.stream
319+
320+
expected = f.read()
321+
322+
assert len(obj.stream) == len(expected)
323+
assert obj.stream == expected

tests/functional/test_input_validations.py

+9-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ def font_samples():
4747

4848

4949
def test_validate_constructor_inputs(pdf_samples, template_stream):
50-
bad_inputs = ["", "True", 1, "12", ("0", "0", "0"), "0", "0", "100"]
50+
bad_inputs = ["", "True", 1, "12", ("0", "0", "0"), "0", "0", "100", "True"]
5151

5252
try:
5353
PyPDFForm(*bad_inputs)
@@ -65,6 +65,14 @@ def test_validate_constructor_inputs(pdf_samples, template_stream):
6565

6666
bad_inputs[1] = False
6767

68+
try:
69+
PyPDFForm(*bad_inputs)
70+
assert False
71+
except InvalidModeError:
72+
assert True
73+
74+
bad_inputs[8] = False
75+
6876
try:
6977
PyPDFForm(*bad_inputs)
7078
assert False

tests/unit/test_filler.py

+26
Original file line numberDiff line numberDiff line change
@@ -78,6 +78,32 @@ def test_fill(template_stream, data_dict):
7878
)
7979

8080

81+
def test_fill_sejda(sejda_template, sejda_data):
82+
elements = TemplateMiddleware().build_elements(sejda_template, sejda=True)
83+
84+
for k, v in elements.items():
85+
if k in sejda_data:
86+
v.value = sejda_data[k]
87+
88+
if elements[k].type == ElementType.text:
89+
elements[k].font = TextConstants().global_font
90+
elements[k].font_size = TextConstants().global_font_size
91+
elements[k].font_color = TextConstants().global_font_color
92+
elements[k].text_x_offset = TextConstants().global_text_x_offset
93+
elements[k].text_y_offset = TextConstants().global_text_y_offset
94+
elements[k].text_wrap_length = TextConstants().global_text_wrap_length
95+
elements[k].validate_constants()
96+
elements[k].validate_value()
97+
elements[k].validate_text_attributes()
98+
99+
result_stream = Filler().fill(sejda_template, elements, sejda=True)
100+
101+
assert result_stream != template_stream
102+
103+
for element in TemplateCore().iterate_elements(result_stream):
104+
assert element[TemplateConstants().parent_key][TemplateConstants().field_editable_key] == pdfrw.PdfObject(1)
105+
106+
81107
def test_fill_with_radiobutton(template_with_radiobutton_stream, data_dict):
82108
elements = TemplateMiddleware().build_elements(template_with_radiobutton_stream)
83109

0 commit comments

Comments
 (0)