@kortix/sandbox 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (246) hide show
  1. package/config/customize.sh +143 -0
  2. package/config/kortix-env-setup.sh +25 -0
  3. package/kortix-master/package.json +22 -0
  4. package/kortix-master/src/config.ts +22 -0
  5. package/kortix-master/src/index.ts +44 -0
  6. package/kortix-master/src/routes/env.ts +65 -0
  7. package/kortix-master/src/routes/proxy.ts +108 -0
  8. package/kortix-master/src/routes/update.ts +185 -0
  9. package/kortix-master/src/services/proxy.ts +43 -0
  10. package/kortix-master/src/services/secret-store.ts +156 -0
  11. package/kortix-master/tsconfig.json +14 -0
  12. package/opencode/agents/kortix-browser.md +142 -0
  13. package/opencode/agents/kortix-build.md +62 -0
  14. package/opencode/agents/kortix-explore.md +66 -0
  15. package/opencode/agents/kortix-image-gen.md +33 -0
  16. package/opencode/agents/kortix-main.md +450 -0
  17. package/opencode/agents/kortix-plan.md +100 -0
  18. package/opencode/agents/kortix-research.md +84 -0
  19. package/opencode/agents/kortix-sheets.md +61 -0
  20. package/opencode/agents/kortix-slides.md +64 -0
  21. package/opencode/agents/kortix-web-dev.md +572 -0
  22. package/opencode/commands/email.md +36 -0
  23. package/opencode/commands/init.md +43 -0
  24. package/opencode/commands/journal.md +44 -0
  25. package/opencode/commands/memory-init.md +81 -0
  26. package/opencode/commands/memory-search.md +50 -0
  27. package/opencode/commands/memory-status.md +56 -0
  28. package/opencode/commands/research.md +36 -0
  29. package/opencode/commands/search.md +38 -0
  30. package/opencode/commands/slides.md +32 -0
  31. package/opencode/commands/spreadsheet.md +30 -0
  32. package/opencode/memory.json +37 -0
  33. package/opencode/ocx.jsonc +10 -0
  34. package/opencode/opencode.jsonc +103 -0
  35. package/opencode/package.json +25 -0
  36. package/opencode/patches/apply.sh +19 -0
  37. package/opencode/patches/opencode-pty-spawn.txt +49 -0
  38. package/opencode/plugin/background-agents.ts.disabled +483 -0
  39. package/opencode/plugin/kdco-primitives/get-project-id.ts +172 -0
  40. package/opencode/plugin/kdco-primitives/index.ts +26 -0
  41. package/opencode/plugin/kdco-primitives/log-warn.ts +51 -0
  42. package/opencode/plugin/kdco-primitives/mutex.ts +122 -0
  43. package/opencode/plugin/kdco-primitives/shell.ts +138 -0
  44. package/opencode/plugin/kdco-primitives/temp.ts +36 -0
  45. package/opencode/plugin/kdco-primitives/terminal-detect.ts +34 -0
  46. package/opencode/plugin/kdco-primitives/types.ts +13 -0
  47. package/opencode/plugin/kdco-primitives/with-timeout.ts +84 -0
  48. package/opencode/plugin/memory.ts +306 -0
  49. package/opencode/plugin/worktree/state.ts +412 -0
  50. package/opencode/plugin/worktree/terminal.ts +1002 -0
  51. package/opencode/plugin/worktree.ts +861 -0
  52. package/opencode/skills/KORTIX-browser/SKILL.md +478 -0
  53. package/opencode/skills/KORTIX-cron-triggers/SKILL.md +173 -0
  54. package/opencode/skills/KORTIX-deep-research/SKILL.md +278 -0
  55. package/opencode/skills/KORTIX-docx/SKILL.md +398 -0
  56. package/opencode/skills/KORTIX-docx/scripts/__init__.py +1 -0
  57. package/opencode/skills/KORTIX-docx/scripts/accept_changes.py +104 -0
  58. package/opencode/skills/KORTIX-docx/scripts/comment.py +244 -0
  59. package/opencode/skills/KORTIX-docx/scripts/office/helpers/__init__.py +0 -0
  60. package/opencode/skills/KORTIX-docx/scripts/office/helpers/merge_runs.py +199 -0
  61. package/opencode/skills/KORTIX-docx/scripts/office/helpers/simplify_redlines.py +197 -0
  62. package/opencode/skills/KORTIX-docx/scripts/office/pack.py +159 -0
  63. package/opencode/skills/KORTIX-docx/scripts/office/soffice.py +183 -0
  64. package/opencode/skills/KORTIX-docx/scripts/office/unpack.py +132 -0
  65. package/opencode/skills/KORTIX-docx/scripts/office/validate.py +111 -0
  66. package/opencode/skills/KORTIX-docx/scripts/office/validators/__init__.py +15 -0
  67. package/opencode/skills/KORTIX-docx/scripts/office/validators/base.py +847 -0
  68. package/opencode/skills/KORTIX-docx/scripts/office/validators/docx.py +446 -0
  69. package/opencode/skills/KORTIX-docx/scripts/office/validators/pptx.py +275 -0
  70. package/opencode/skills/KORTIX-docx/scripts/office/validators/redlining.py +247 -0
  71. package/opencode/skills/KORTIX-docx/scripts/render_docx.py +179 -0
  72. package/opencode/skills/KORTIX-docx/scripts/templates/comments.xml +3 -0
  73. package/opencode/skills/KORTIX-docx/scripts/templates/commentsExtended.xml +3 -0
  74. package/opencode/skills/KORTIX-docx/scripts/templates/commentsExtensible.xml +3 -0
  75. package/opencode/skills/KORTIX-docx/scripts/templates/commentsIds.xml +3 -0
  76. package/opencode/skills/KORTIX-docx/scripts/templates/people.xml +3 -0
  77. package/opencode/skills/KORTIX-domain-research/SKILL.md +96 -0
  78. package/opencode/skills/KORTIX-domain-research/scripts/domain-lookup.py +810 -0
  79. package/opencode/skills/KORTIX-elevenlabs/SKILL.md +230 -0
  80. package/opencode/skills/KORTIX-elevenlabs/scripts/tts.py +389 -0
  81. package/opencode/skills/KORTIX-email/SKILL.md +145 -0
  82. package/opencode/skills/KORTIX-legal-writer/SKILL.md +409 -0
  83. package/opencode/skills/KORTIX-legal-writer/references/bluebook.md +152 -0
  84. package/opencode/skills/KORTIX-legal-writer/references/document-types.md +416 -0
  85. package/opencode/skills/KORTIX-legal-writer/scripts/courtlistener.py +291 -0
  86. package/opencode/skills/KORTIX-legal-writer/scripts/ecfr_lookup.py +299 -0
  87. package/opencode/skills/KORTIX-legal-writer/scripts/verify-legal.py +507 -0
  88. package/opencode/skills/KORTIX-logo-creator/SKILL.md +293 -0
  89. package/opencode/skills/KORTIX-logo-creator/references/prompt-patterns.md +134 -0
  90. package/opencode/skills/KORTIX-logo-creator/scripts/compose_logo.py +406 -0
  91. package/opencode/skills/KORTIX-logo-creator/scripts/create_logo_sheet.py +258 -0
  92. package/opencode/skills/KORTIX-logo-creator/scripts/remove_bg.py +96 -0
  93. package/opencode/skills/KORTIX-memory/SKILL.md +261 -0
  94. package/opencode/skills/KORTIX-memory/scripts/export-sessions.py +409 -0
  95. package/opencode/skills/KORTIX-paper-creator/SKILL.md +549 -0
  96. package/opencode/skills/KORTIX-paper-creator/assets/template.tex +101 -0
  97. package/opencode/skills/KORTIX-paper-creator/scripts/compile.sh +177 -0
  98. package/opencode/skills/KORTIX-paper-creator/scripts/openalex_to_bibtex.py +220 -0
  99. package/opencode/skills/KORTIX-paper-creator/scripts/verify.sh +354 -0
  100. package/opencode/skills/KORTIX-paper-search/SKILL.md +418 -0
  101. package/opencode/skills/KORTIX-pdf/SKILL.md +232 -0
  102. package/opencode/skills/KORTIX-pdf/forms.md +36 -0
  103. package/opencode/skills/KORTIX-pdf/reference.md +105 -0
  104. package/opencode/skills/KORTIX-pdf/scripts/check_bounding_boxes.py +65 -0
  105. package/opencode/skills/KORTIX-pdf/scripts/check_fillable_fields.py +11 -0
  106. package/opencode/skills/KORTIX-pdf/scripts/convert_pdf_to_images.py +33 -0
  107. package/opencode/skills/KORTIX-pdf/scripts/create_validation_image.py +37 -0
  108. package/opencode/skills/KORTIX-pdf/scripts/extract_form_field_info.py +122 -0
  109. package/opencode/skills/KORTIX-pdf/scripts/extract_form_structure.py +115 -0
  110. package/opencode/skills/KORTIX-pdf/scripts/fill_fillable_fields.py +98 -0
  111. package/opencode/skills/KORTIX-pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
  112. package/opencode/skills/KORTIX-plan/SKILL.md +228 -0
  113. package/opencode/skills/KORTIX-presentation-viewer/SKILL.md +87 -0
  114. package/opencode/skills/KORTIX-presentation-viewer/serve.ts +136 -0
  115. package/opencode/skills/KORTIX-presentation-viewer/viewer.html +559 -0
  116. package/opencode/skills/KORTIX-presentations/SKILL.md +344 -0
  117. package/opencode/skills/KORTIX-remotion/SKILL.md +56 -0
  118. package/opencode/skills/KORTIX-remotion/rules/3d.md +86 -0
  119. package/opencode/skills/KORTIX-remotion/rules/animations.md +29 -0
  120. package/opencode/skills/KORTIX-remotion/rules/assets.md +78 -0
  121. package/opencode/skills/KORTIX-remotion/rules/audio-visualization.md +198 -0
  122. package/opencode/skills/KORTIX-remotion/rules/audio.md +169 -0
  123. package/opencode/skills/KORTIX-remotion/rules/calculate-metadata.md +104 -0
  124. package/opencode/skills/KORTIX-remotion/rules/can-decode.md +75 -0
  125. package/opencode/skills/KORTIX-remotion/rules/charts.md +120 -0
  126. package/opencode/skills/KORTIX-remotion/rules/compositions.md +141 -0
  127. package/opencode/skills/KORTIX-remotion/rules/display-captions.md +184 -0
  128. package/opencode/skills/KORTIX-remotion/rules/extract-frames.md +229 -0
  129. package/opencode/skills/KORTIX-remotion/rules/ffmpeg.md +38 -0
  130. package/opencode/skills/KORTIX-remotion/rules/fonts.md +152 -0
  131. package/opencode/skills/KORTIX-remotion/rules/get-audio-duration.md +58 -0
  132. package/opencode/skills/KORTIX-remotion/rules/get-video-dimensions.md +68 -0
  133. package/opencode/skills/KORTIX-remotion/rules/get-video-duration.md +58 -0
  134. package/opencode/skills/KORTIX-remotion/rules/gifs.md +141 -0
  135. package/opencode/skills/KORTIX-remotion/rules/images.md +130 -0
  136. package/opencode/skills/KORTIX-remotion/rules/import-srt-captions.md +69 -0
  137. package/opencode/skills/KORTIX-remotion/rules/light-leaks.md +73 -0
  138. package/opencode/skills/KORTIX-remotion/rules/lottie.md +68 -0
  139. package/opencode/skills/KORTIX-remotion/rules/maps.md +401 -0
  140. package/opencode/skills/KORTIX-remotion/rules/measuring-dom-nodes.md +35 -0
  141. package/opencode/skills/KORTIX-remotion/rules/measuring-text.md +143 -0
  142. package/opencode/skills/KORTIX-remotion/rules/parameters.md +98 -0
  143. package/opencode/skills/KORTIX-remotion/rules/sequencing.md +118 -0
  144. package/opencode/skills/KORTIX-remotion/rules/subtitles.md +36 -0
  145. package/opencode/skills/KORTIX-remotion/rules/tailwind.md +11 -0
  146. package/opencode/skills/KORTIX-remotion/rules/text-animations.md +20 -0
  147. package/opencode/skills/KORTIX-remotion/rules/timing.md +179 -0
  148. package/opencode/skills/KORTIX-remotion/rules/transcribe-captions.md +70 -0
  149. package/opencode/skills/KORTIX-remotion/rules/transitions.md +197 -0
  150. package/opencode/skills/KORTIX-remotion/rules/transparent-videos.md +106 -0
  151. package/opencode/skills/KORTIX-remotion/rules/trimming.md +53 -0
  152. package/opencode/skills/KORTIX-remotion/rules/videos.md +171 -0
  153. package/opencode/skills/KORTIX-secrets/SKILL.md +280 -0
  154. package/opencode/skills/KORTIX-semantic-search/SKILL.md +213 -0
  155. package/opencode/skills/KORTIX-session-search/SKILL.md +807 -0
  156. package/opencode/skills/KORTIX-session-search/Untitled +1 -0
  157. package/opencode/skills/KORTIX-skill-creator/SKILL.md +163 -0
  158. package/opencode/skills/KORTIX-web-research/SKILL.md +69 -0
  159. package/opencode/skills/KORTIX-xlsx/LICENSE.txt +30 -0
  160. package/opencode/skills/KORTIX-xlsx/SKILL.md +549 -0
  161. package/opencode/skills/KORTIX-xlsx/scripts/office/helpers/__init__.py +0 -0
  162. package/opencode/skills/KORTIX-xlsx/scripts/office/helpers/merge_runs.py +199 -0
  163. package/opencode/skills/KORTIX-xlsx/scripts/office/helpers/simplify_redlines.py +197 -0
  164. package/opencode/skills/KORTIX-xlsx/scripts/office/pack.py +159 -0
  165. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  166. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  167. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  168. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  169. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  170. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  171. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  172. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  173. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  174. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  175. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  176. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  177. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  178. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  179. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  180. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  181. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  182. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  183. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  184. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  185. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  186. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  187. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  188. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  189. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  190. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  191. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  192. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  193. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  194. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  195. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  196. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/mce/mc.xsd +75 -0
  197. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
  198. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
  199. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
  200. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
  201. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
  202. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  203. package/opencode/skills/KORTIX-xlsx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
  204. package/opencode/skills/KORTIX-xlsx/scripts/office/soffice.py +183 -0
  205. package/opencode/skills/KORTIX-xlsx/scripts/office/unpack.py +132 -0
  206. package/opencode/skills/KORTIX-xlsx/scripts/office/validate.py +111 -0
  207. package/opencode/skills/KORTIX-xlsx/scripts/office/validators/__init__.py +15 -0
  208. package/opencode/skills/KORTIX-xlsx/scripts/office/validators/base.py +847 -0
  209. package/opencode/skills/KORTIX-xlsx/scripts/office/validators/docx.py +446 -0
  210. package/opencode/skills/KORTIX-xlsx/scripts/office/validators/pptx.py +275 -0
  211. package/opencode/skills/KORTIX-xlsx/scripts/office/validators/redlining.py +247 -0
  212. package/opencode/skills/KORTIX-xlsx/scripts/recalc.py +184 -0
  213. package/opencode/tools/image-gen.ts +342 -0
  214. package/opencode/tools/image-search.ts +190 -0
  215. package/opencode/tools/memory-get.ts +168 -0
  216. package/opencode/tools/memory-search.ts +247 -0
  217. package/opencode/tools/presentation-gen.ts +723 -0
  218. package/opencode/tools/scrape-webpage.ts +115 -0
  219. package/opencode/tools/scripts/.python-version +1 -0
  220. package/opencode/tools/scripts/convert_pdf.py +184 -0
  221. package/opencode/tools/scripts/convert_pptx.py +562 -0
  222. package/opencode/tools/scripts/pyproject.toml +11 -0
  223. package/opencode/tools/scripts/uv.lock +287 -0
  224. package/opencode/tools/scripts/validate_slide.py +74 -0
  225. package/opencode/tools/show-user.ts +217 -0
  226. package/opencode/tools/tests/e2e-presentation-fix.ts +277 -0
  227. package/opencode/tools/tests/image-gen.test.ts +215 -0
  228. package/opencode/tools/tests/image-search.test.ts +125 -0
  229. package/opencode/tools/tests/memory-system-benchmark.ts +1076 -0
  230. package/opencode/tools/tests/presentation-gen.test.ts +389 -0
  231. package/opencode/tools/tests/scrape-webpage.test.ts +74 -0
  232. package/opencode/tools/tests/show-user.test.ts +241 -0
  233. package/opencode/tools/tests/video-gen.test.ts +110 -0
  234. package/opencode/tools/tests/web-search.test.ts +106 -0
  235. package/opencode/tools/video-gen.ts +200 -0
  236. package/opencode/tools/web-search.ts +153 -0
  237. package/opencode/tsconfig.json +29 -0
  238. package/package.json +36 -0
  239. package/patch-agent-browser.js +100 -0
  240. package/postinstall.sh +88 -0
  241. package/services/KORTIX-presentation-viewer/run +37 -0
  242. package/services/agent-browser-viewer/run +48 -0
  243. package/services/kortix-master/run +16 -0
  244. package/services/lss-sync/run +22 -0
  245. package/services/opencode-serve/run +25 -0
  246. package/services/opencode-web/run +21 -0
@@ -0,0 +1,105 @@
1
+ # PDF Processing Advanced Reference
2
+
3
+ ## pypdfium2 Library
4
+
5
+ ### Render PDF to Images
6
+ ```python
7
+ import pypdfium2 as pdfium
8
+
9
+ pdf = pdfium.PdfDocument("document.pdf")
10
+ page = pdf[0]
11
+ bitmap = page.render(scale=2.0, rotation=0)
12
+ img = bitmap.to_pil()
13
+ img.save("page_1.png", "PNG")
14
+
15
+ for i, page in enumerate(pdf):
16
+ bitmap = page.render(scale=1.5)
17
+ img = bitmap.to_pil()
18
+ img.save(f"page_{i+1}.jpg", "JPEG", quality=90)
19
+ ```
20
+
21
+ ## JavaScript Libraries
22
+
23
+ ### pdf-lib (MIT License)
24
+
25
+ #### Load and Manipulate
26
+ ```javascript
27
+ import { PDFDocument } from 'pdf-lib';
28
+ import fs from 'fs';
29
+
30
+ async function manipulatePDF() {
31
+ const existingPdfBytes = fs.readFileSync('input.pdf');
32
+ const pdfDoc = await PDFDocument.load(existingPdfBytes);
33
+ const newPage = pdfDoc.addPage([600, 400]);
34
+ newPage.drawText('Added by pdf-lib', { x: 100, y: 300, size: 16 });
35
+ const pdfBytes = await pdfDoc.save();
36
+ fs.writeFileSync('modified.pdf', pdfBytes);
37
+ }
38
+ ```
39
+
40
+ #### Create from Scratch
41
+ ```javascript
42
+ import { PDFDocument, rgb, StandardFonts } from 'pdf-lib';
43
+
44
+ async function createPDF() {
45
+ const pdfDoc = await PDFDocument.create();
46
+ const font = await pdfDoc.embedFont(StandardFonts.HelveticaBold);
47
+ const page = pdfDoc.addPage([595, 842]);
48
+ page.drawText('Invoice #12345', { x: 50, y: page.getSize().height - 50, size: 18, font, color: rgb(0.2, 0.2, 0.8) });
49
+ fs.writeFileSync('created.pdf', await pdfDoc.save());
50
+ }
51
+ ```
52
+
53
+ #### Merge
54
+ ```javascript
55
+ async function mergePDFs() {
56
+ const mergedPdf = await PDFDocument.create();
57
+ const pdf1 = await PDFDocument.load(fs.readFileSync('doc1.pdf'));
58
+ const pdf2 = await PDFDocument.load(fs.readFileSync('doc2.pdf'));
59
+ const pdf1Pages = await mergedPdf.copyPages(pdf1, pdf1.getPageIndices());
60
+ pdf1Pages.forEach(page => mergedPdf.addPage(page));
61
+ const pdf2Pages = await mergedPdf.copyPages(pdf2, [0, 2, 4]);
62
+ pdf2Pages.forEach(page => mergedPdf.addPage(page));
63
+ fs.writeFileSync('merged.pdf', await mergedPdf.save());
64
+ }
65
+ ```
66
+
67
+ ## Advanced CLI
68
+
69
+ ### poppler-utils
70
+ ```bash
71
+ pdftotext -bbox-layout document.pdf output.xml
72
+ pdftoppm -png -r 300 document.pdf output_prefix
73
+ pdfimages -j -p document.pdf page_images
74
+ ```
75
+
76
+ ### qpdf
77
+ ```bash
78
+ qpdf --split-pages=3 input.pdf output_group_%02d.pdf
79
+ qpdf --linearize input.pdf optimized.pdf
80
+ qpdf --check input.pdf
81
+ qpdf --encrypt user_pass owner_pass 256 --print=none --modify=none -- input.pdf encrypted.pdf
82
+ ```
83
+
84
+ ## Advanced pdfplumber
85
+
86
+ ```python
87
+ import pdfplumber
88
+
89
+ with pdfplumber.open("document.pdf") as pdf:
90
+ page = pdf.pages[0]
91
+ chars = page.chars # text with coordinates
92
+ bbox_text = page.within_bbox((100, 100, 400, 200)).extract_text()
93
+ tables = page.extract_tables({
94
+ "vertical_strategy": "lines",
95
+ "horizontal_strategy": "lines",
96
+ "snap_tolerance": 3,
97
+ })
98
+ ```
99
+
100
+ ## Performance Tips
101
+
102
+ - Use `pdftotext -bbox-layout` for fastest text extraction
103
+ - Use `pdfimages` over page rendering for image extraction
104
+ - Process large PDFs in chunks
105
+ - Use `qpdf --split-pages` for splitting large files
@@ -0,0 +1,65 @@
1
+ from dataclasses import dataclass
2
+ import json
3
+ import sys
4
+
5
+
6
+
7
+
8
+ @dataclass
9
+ class RectAndField:
10
+ rect: list[float]
11
+ rect_type: str
12
+ field: dict
13
+
14
+
15
+ def get_bounding_box_messages(fields_json_stream) -> list[str]:
16
+ messages = []
17
+ fields = json.load(fields_json_stream)
18
+ messages.append(f"Read {len(fields['form_fields'])} fields")
19
+
20
+ def rects_intersect(r1, r2):
21
+ disjoint_horizontal = r1[0] >= r2[2] or r1[2] <= r2[0]
22
+ disjoint_vertical = r1[1] >= r2[3] or r1[3] <= r2[1]
23
+ return not (disjoint_horizontal or disjoint_vertical)
24
+
25
+ rects_and_fields = []
26
+ for f in fields["form_fields"]:
27
+ rects_and_fields.append(RectAndField(f["label_bounding_box"], "label", f))
28
+ rects_and_fields.append(RectAndField(f["entry_bounding_box"], "entry", f))
29
+
30
+ has_error = False
31
+ for i, ri in enumerate(rects_and_fields):
32
+ for j in range(i + 1, len(rects_and_fields)):
33
+ rj = rects_and_fields[j]
34
+ if ri.field["page_number"] == rj.field["page_number"] and rects_intersect(ri.rect, rj.rect):
35
+ has_error = True
36
+ if ri.field is rj.field:
37
+ messages.append(f"FAILURE: intersection between label and entry bounding boxes for `{ri.field['description']}` ({ri.rect}, {rj.rect})")
38
+ else:
39
+ messages.append(f"FAILURE: intersection between {ri.rect_type} bounding box for `{ri.field['description']}` ({ri.rect}) and {rj.rect_type} bounding box for `{rj.field['description']}` ({rj.rect})")
40
+ if len(messages) >= 20:
41
+ messages.append("Aborting further checks; fix bounding boxes and try again")
42
+ return messages
43
+ if ri.rect_type == "entry":
44
+ if "entry_text" in ri.field:
45
+ font_size = ri.field["entry_text"].get("font_size", 14)
46
+ entry_height = ri.rect[3] - ri.rect[1]
47
+ if entry_height < font_size:
48
+ has_error = True
49
+ messages.append(f"FAILURE: entry bounding box height ({entry_height}) for `{ri.field['description']}` is too short for the text content (font size: {font_size}). Increase the box height or decrease the font size.")
50
+ if len(messages) >= 20:
51
+ messages.append("Aborting further checks; fix bounding boxes and try again")
52
+ return messages
53
+
54
+ if not has_error:
55
+ messages.append("SUCCESS: All bounding boxes are valid")
56
+ return messages
57
+
58
+ if __name__ == "__main__":
59
+ if len(sys.argv) != 2:
60
+ print("Usage: check_bounding_boxes.py [fields.json]")
61
+ sys.exit(1)
62
+ with open(sys.argv[1]) as f:
63
+ messages = get_bounding_box_messages(f)
64
+ for msg in messages:
65
+ print(msg)
@@ -0,0 +1,11 @@
1
+ import sys
2
+ from pypdf import PdfReader
3
+
4
+
5
+
6
+
7
+ reader = PdfReader(sys.argv[1])
8
+ if (reader.get_fields()):
9
+ print("This PDF has fillable form fields")
10
+ else:
11
+ print("This PDF does not have fillable form fields; you will need to visually determine where to enter data")
@@ -0,0 +1,33 @@
1
+ import os
2
+ import sys
3
+
4
+ from pdf2image import convert_from_path
5
+
6
+
7
+
8
+
9
+ def convert(pdf_path, output_dir, max_dim=1000):
10
+ images = convert_from_path(pdf_path, dpi=200)
11
+
12
+ for i, image in enumerate(images):
13
+ width, height = image.size
14
+ if width > max_dim or height > max_dim:
15
+ scale_factor = min(max_dim / width, max_dim / height)
16
+ new_width = int(width * scale_factor)
17
+ new_height = int(height * scale_factor)
18
+ image = image.resize((new_width, new_height))
19
+
20
+ image_path = os.path.join(output_dir, f"page_{i+1}.png")
21
+ image.save(image_path)
22
+ print(f"Saved page {i+1} as {image_path} (size: {image.size})")
23
+
24
+ print(f"Converted {len(images)} pages to PNG images")
25
+
26
+
27
+ if __name__ == "__main__":
28
+ if len(sys.argv) != 3:
29
+ print("Usage: convert_pdf_to_images.py [input pdf] [output directory]")
30
+ sys.exit(1)
31
+ pdf_path = sys.argv[1]
32
+ output_directory = sys.argv[2]
33
+ convert(pdf_path, output_directory)
@@ -0,0 +1,37 @@
1
+ import json
2
+ import sys
3
+
4
+ from PIL import Image, ImageDraw
5
+
6
+
7
+
8
+
9
+ def create_validation_image(page_number, fields_json_path, input_path, output_path):
10
+ with open(fields_json_path, 'r') as f:
11
+ data = json.load(f)
12
+
13
+ img = Image.open(input_path)
14
+ draw = ImageDraw.Draw(img)
15
+ num_boxes = 0
16
+
17
+ for field in data["form_fields"]:
18
+ if field["page_number"] == page_number:
19
+ entry_box = field['entry_bounding_box']
20
+ label_box = field['label_bounding_box']
21
+ draw.rectangle(entry_box, outline='red', width=2)
22
+ draw.rectangle(label_box, outline='blue', width=2)
23
+ num_boxes += 2
24
+
25
+ img.save(output_path)
26
+ print(f"Created validation image at {output_path} with {num_boxes} bounding boxes")
27
+
28
+
29
+ if __name__ == "__main__":
30
+ if len(sys.argv) != 5:
31
+ print("Usage: create_validation_image.py [page number] [fields.json file] [input image path] [output image path]")
32
+ sys.exit(1)
33
+ page_number = int(sys.argv[1])
34
+ fields_json_path = sys.argv[2]
35
+ input_image_path = sys.argv[3]
36
+ output_image_path = sys.argv[4]
37
+ create_validation_image(page_number, fields_json_path, input_image_path, output_image_path)
@@ -0,0 +1,122 @@
1
+ import json
2
+ import sys
3
+
4
+ from pypdf import PdfReader
5
+
6
+
7
+
8
+
9
+ def get_full_annotation_field_id(annotation):
10
+ components = []
11
+ while annotation:
12
+ field_name = annotation.get('/T')
13
+ if field_name:
14
+ components.append(field_name)
15
+ annotation = annotation.get('/Parent')
16
+ return ".".join(reversed(components)) if components else None
17
+
18
+
19
+ def make_field_dict(field, field_id):
20
+ field_dict = {"field_id": field_id}
21
+ ft = field.get('/FT')
22
+ if ft == "/Tx":
23
+ field_dict["type"] = "text"
24
+ elif ft == "/Btn":
25
+ field_dict["type"] = "checkbox"
26
+ states = field.get("/_States_", [])
27
+ if len(states) == 2:
28
+ if "/Off" in states:
29
+ field_dict["checked_value"] = states[0] if states[0] != "/Off" else states[1]
30
+ field_dict["unchecked_value"] = "/Off"
31
+ else:
32
+ print(f"Unexpected state values for checkbox `${field_id}`. Its checked and unchecked values may not be correct; if you're trying to check it, visually verify the results.")
33
+ field_dict["checked_value"] = states[0]
34
+ field_dict["unchecked_value"] = states[1]
35
+ elif ft == "/Ch":
36
+ field_dict["type"] = "choice"
37
+ states = field.get("/_States_", [])
38
+ field_dict["choice_options"] = [{
39
+ "value": state[0],
40
+ "text": state[1],
41
+ } for state in states]
42
+ else:
43
+ field_dict["type"] = f"unknown ({ft})"
44
+ return field_dict
45
+
46
+
47
+ def get_field_info(reader: PdfReader):
48
+ fields = reader.get_fields()
49
+
50
+ field_info_by_id = {}
51
+ possible_radio_names = set()
52
+
53
+ for field_id, field in fields.items():
54
+ if field.get("/Kids"):
55
+ if field.get("/FT") == "/Btn":
56
+ possible_radio_names.add(field_id)
57
+ continue
58
+ field_info_by_id[field_id] = make_field_dict(field, field_id)
59
+
60
+
61
+ radio_fields_by_id = {}
62
+
63
+ for page_index, page in enumerate(reader.pages):
64
+ annotations = page.get('/Annots', [])
65
+ for ann in annotations:
66
+ field_id = get_full_annotation_field_id(ann)
67
+ if field_id in field_info_by_id:
68
+ field_info_by_id[field_id]["page"] = page_index + 1
69
+ field_info_by_id[field_id]["rect"] = ann.get('/Rect')
70
+ elif field_id in possible_radio_names:
71
+ try:
72
+ on_values = [v for v in ann["/AP"]["/N"] if v != "/Off"]
73
+ except KeyError:
74
+ continue
75
+ if len(on_values) == 1:
76
+ rect = ann.get("/Rect")
77
+ if field_id not in radio_fields_by_id:
78
+ radio_fields_by_id[field_id] = {
79
+ "field_id": field_id,
80
+ "type": "radio_group",
81
+ "page": page_index + 1,
82
+ "radio_options": [],
83
+ }
84
+ radio_fields_by_id[field_id]["radio_options"].append({
85
+ "value": on_values[0],
86
+ "rect": rect,
87
+ })
88
+
89
+ fields_with_location = []
90
+ for field_info in field_info_by_id.values():
91
+ if "page" in field_info:
92
+ fields_with_location.append(field_info)
93
+ else:
94
+ print(f"Unable to determine location for field id: {field_info.get('field_id')}, ignoring")
95
+
96
+ def sort_key(f):
97
+ if "radio_options" in f:
98
+ rect = f["radio_options"][0]["rect"] or [0, 0, 0, 0]
99
+ else:
100
+ rect = f.get("rect") or [0, 0, 0, 0]
101
+ adjusted_position = [-rect[1], rect[0]]
102
+ return [f.get("page"), adjusted_position]
103
+
104
+ sorted_fields = fields_with_location + list(radio_fields_by_id.values())
105
+ sorted_fields.sort(key=sort_key)
106
+
107
+ return sorted_fields
108
+
109
+
110
+ def write_field_info(pdf_path: str, json_output_path: str):
111
+ reader = PdfReader(pdf_path)
112
+ field_info = get_field_info(reader)
113
+ with open(json_output_path, "w") as f:
114
+ json.dump(field_info, f, indent=2)
115
+ print(f"Wrote {len(field_info)} fields to {json_output_path}")
116
+
117
+
118
+ if __name__ == "__main__":
119
+ if len(sys.argv) != 3:
120
+ print("Usage: extract_form_field_info.py [input pdf] [output json]")
121
+ sys.exit(1)
122
+ write_field_info(sys.argv[1], sys.argv[2])
@@ -0,0 +1,115 @@
1
+ """
2
+ Extract form structure from a non-fillable PDF.
3
+
4
+ This script analyzes the PDF to find:
5
+ - Text labels with their exact coordinates
6
+ - Horizontal lines (row boundaries)
7
+ - Checkboxes (small rectangles)
8
+
9
+ Output: A JSON file with the form structure that can be used to generate
10
+ accurate field coordinates for filling.
11
+
12
+ Usage: python extract_form_structure.py <input.pdf> <output.json>
13
+ """
14
+
15
+ import json
16
+ import sys
17
+ import pdfplumber
18
+
19
+
20
+ def extract_form_structure(pdf_path):
21
+ structure = {
22
+ "pages": [],
23
+ "labels": [],
24
+ "lines": [],
25
+ "checkboxes": [],
26
+ "row_boundaries": []
27
+ }
28
+
29
+ with pdfplumber.open(pdf_path) as pdf:
30
+ for page_num, page in enumerate(pdf.pages, 1):
31
+ structure["pages"].append({
32
+ "page_number": page_num,
33
+ "width": float(page.width),
34
+ "height": float(page.height)
35
+ })
36
+
37
+ words = page.extract_words()
38
+ for word in words:
39
+ structure["labels"].append({
40
+ "page": page_num,
41
+ "text": word["text"],
42
+ "x0": round(float(word["x0"]), 1),
43
+ "top": round(float(word["top"]), 1),
44
+ "x1": round(float(word["x1"]), 1),
45
+ "bottom": round(float(word["bottom"]), 1)
46
+ })
47
+
48
+ for line in page.lines:
49
+ if abs(float(line["x1"]) - float(line["x0"])) > page.width * 0.5:
50
+ structure["lines"].append({
51
+ "page": page_num,
52
+ "y": round(float(line["top"]), 1),
53
+ "x0": round(float(line["x0"]), 1),
54
+ "x1": round(float(line["x1"]), 1)
55
+ })
56
+
57
+ for rect in page.rects:
58
+ width = float(rect["x1"]) - float(rect["x0"])
59
+ height = float(rect["bottom"]) - float(rect["top"])
60
+ if 5 <= width <= 15 and 5 <= height <= 15 and abs(width - height) < 2:
61
+ structure["checkboxes"].append({
62
+ "page": page_num,
63
+ "x0": round(float(rect["x0"]), 1),
64
+ "top": round(float(rect["top"]), 1),
65
+ "x1": round(float(rect["x1"]), 1),
66
+ "bottom": round(float(rect["bottom"]), 1),
67
+ "center_x": round((float(rect["x0"]) + float(rect["x1"])) / 2, 1),
68
+ "center_y": round((float(rect["top"]) + float(rect["bottom"])) / 2, 1)
69
+ })
70
+
71
+ lines_by_page = {}
72
+ for line in structure["lines"]:
73
+ page = line["page"]
74
+ if page not in lines_by_page:
75
+ lines_by_page[page] = []
76
+ lines_by_page[page].append(line["y"])
77
+
78
+ for page, y_coords in lines_by_page.items():
79
+ y_coords = sorted(set(y_coords))
80
+ for i in range(len(y_coords) - 1):
81
+ structure["row_boundaries"].append({
82
+ "page": page,
83
+ "row_top": y_coords[i],
84
+ "row_bottom": y_coords[i + 1],
85
+ "row_height": round(y_coords[i + 1] - y_coords[i], 1)
86
+ })
87
+
88
+ return structure
89
+
90
+
91
+ def main():
92
+ if len(sys.argv) != 3:
93
+ print("Usage: extract_form_structure.py <input.pdf> <output.json>")
94
+ sys.exit(1)
95
+
96
+ pdf_path = sys.argv[1]
97
+ output_path = sys.argv[2]
98
+
99
+ print(f"Extracting structure from {pdf_path}...")
100
+ structure = extract_form_structure(pdf_path)
101
+
102
+ with open(output_path, "w") as f:
103
+ json.dump(structure, f, indent=2)
104
+
105
+ print(f"Found:")
106
+ print(f" - {len(structure['pages'])} pages")
107
+ print(f" - {len(structure['labels'])} text labels")
108
+ print(f" - {len(structure['lines'])} horizontal lines")
109
+ print(f" - {len(structure['checkboxes'])} checkboxes")
110
+ print(f" - {len(structure['row_boundaries'])} row boundaries")
111
+ print(f"Saved to {output_path}")
112
+
113
+
114
+ if __name__ == "__main__":
115
+ main()
@@ -0,0 +1,98 @@
1
+ import json
2
+ import sys
3
+
4
+ from pypdf import PdfReader, PdfWriter
5
+
6
+ from extract_form_field_info import get_field_info
7
+
8
+
9
+
10
+
11
+ def fill_pdf_fields(input_pdf_path: str, fields_json_path: str, output_pdf_path: str):
12
+ with open(fields_json_path) as f:
13
+ fields = json.load(f)
14
+ fields_by_page = {}
15
+ for field in fields:
16
+ if "value" in field:
17
+ field_id = field["field_id"]
18
+ page = field["page"]
19
+ if page not in fields_by_page:
20
+ fields_by_page[page] = {}
21
+ fields_by_page[page][field_id] = field["value"]
22
+
23
+ reader = PdfReader(input_pdf_path)
24
+
25
+ has_error = False
26
+ field_info = get_field_info(reader)
27
+ fields_by_ids = {f["field_id"]: f for f in field_info}
28
+ for field in fields:
29
+ existing_field = fields_by_ids.get(field["field_id"])
30
+ if not existing_field:
31
+ has_error = True
32
+ print(f"ERROR: `{field['field_id']}` is not a valid field ID")
33
+ elif field["page"] != existing_field["page"]:
34
+ has_error = True
35
+ print(f"ERROR: Incorrect page number for `{field['field_id']}` (got {field['page']}, expected {existing_field['page']})")
36
+ else:
37
+ if "value" in field:
38
+ err = validation_error_for_field_value(existing_field, field["value"])
39
+ if err:
40
+ print(err)
41
+ has_error = True
42
+ if has_error:
43
+ sys.exit(1)
44
+
45
+ writer = PdfWriter(clone_from=reader)
46
+ for page, field_values in fields_by_page.items():
47
+ writer.update_page_form_field_values(writer.pages[page - 1], field_values, auto_regenerate=False)
48
+
49
+ writer.set_need_appearances_writer(True)
50
+
51
+ with open(output_pdf_path, "wb") as f:
52
+ writer.write(f)
53
+
54
+
55
+ def validation_error_for_field_value(field_info, field_value):
56
+ field_type = field_info["type"]
57
+ field_id = field_info["field_id"]
58
+ if field_type == "checkbox":
59
+ checked_val = field_info["checked_value"]
60
+ unchecked_val = field_info["unchecked_value"]
61
+ if field_value != checked_val and field_value != unchecked_val:
62
+ return f'ERROR: Invalid value "{field_value}" for checkbox field "{field_id}". The checked value is "{checked_val}" and the unchecked value is "{unchecked_val}"'
63
+ elif field_type == "radio_group":
64
+ option_values = [opt["value"] for opt in field_info["radio_options"]]
65
+ if field_value not in option_values:
66
+ return f'ERROR: Invalid value "{field_value}" for radio group field "{field_id}". Valid values are: {option_values}'
67
+ elif field_type == "choice":
68
+ choice_values = [opt["value"] for opt in field_info["choice_options"]]
69
+ if field_value not in choice_values:
70
+ return f'ERROR: Invalid value "{field_value}" for choice field "{field_id}". Valid values are: {choice_values}'
71
+ return None
72
+
73
+
74
+ def monkeypatch_pydpf_method():
75
+ from pypdf.generic import DictionaryObject
76
+ from pypdf.constants import FieldDictionaryAttributes
77
+
78
+ original_get_inherited = DictionaryObject.get_inherited
79
+
80
+ def patched_get_inherited(self, key: str, default = None):
81
+ result = original_get_inherited(self, key, default)
82
+ if key == FieldDictionaryAttributes.Opt:
83
+ if isinstance(result, list) and all(isinstance(v, list) and len(v) == 2 for v in result):
84
+ result = [r[0] for r in result]
85
+ return result
86
+
87
+ DictionaryObject.get_inherited = patched_get_inherited
88
+
89
+
90
+ if __name__ == "__main__":
91
+ if len(sys.argv) != 4:
92
+ print("Usage: fill_fillable_fields.py [input pdf] [field_values.json] [output pdf]")
93
+ sys.exit(1)
94
+ monkeypatch_pydpf_method()
95
+ input_pdf = sys.argv[1]
96
+ fields_json = sys.argv[2]
97
+ output_pdf = sys.argv[3]
98
+ fill_pdf_fields(input_pdf, fields_json, output_pdf)