@things-factory/pdf 7.0.33 → 7.0.37

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 (107) hide show
  1. package/client/bootstrap.js +6 -0
  2. package/client/pages/pdf-release/pdf-release-importer.ts +90 -0
  3. package/client/pages/pdf-release/pdf-release-list-page.ts +398 -0
  4. package/client/pages/pdf-template/pdf-template-importer.ts +86 -0
  5. package/client/pages/pdf-template/pdf-template-list-page.ts +491 -0
  6. package/client/route.ts +11 -0
  7. package/client/tsconfig.json +13 -0
  8. package/dist-client/bootstrap.d.ts +1 -0
  9. package/dist-client/bootstrap.js +6 -0
  10. package/dist-client/bootstrap.js.map +1 -0
  11. package/dist-client/pages/pdf-release/pdf-release-importer.d.ts +23 -0
  12. package/dist-client/pages/pdf-release/pdf-release-importer.js +93 -0
  13. package/dist-client/pages/pdf-release/pdf-release-importer.js.map +1 -0
  14. package/dist-client/pages/pdf-release/pdf-release-list-page.d.ts +66 -0
  15. package/dist-client/pages/pdf-release/pdf-release-list-page.js +370 -0
  16. package/dist-client/pages/pdf-release/pdf-release-list-page.js.map +1 -0
  17. package/dist-client/pages/pdf-template/pdf-template-importer.d.ts +23 -0
  18. package/dist-client/pages/pdf-template/pdf-template-importer.js +93 -0
  19. package/dist-client/pages/pdf-template/pdf-template-importer.js.map +1 -0
  20. package/dist-client/pages/pdf-template/pdf-template-list-page.d.ts +66 -0
  21. package/dist-client/pages/pdf-template/pdf-template-list-page.js +464 -0
  22. package/dist-client/pages/pdf-template/pdf-template-list-page.js.map +1 -0
  23. package/dist-client/route.d.ts +1 -0
  24. package/dist-client/route.js +11 -0
  25. package/dist-client/route.js.map +1 -0
  26. package/dist-client/tsconfig.tsbuildinfo +1 -0
  27. package/dist-server/controller/pdf-service.d.ts +4 -0
  28. package/dist-server/controller/pdf-service.js +34 -0
  29. package/dist-server/controller/pdf-service.js.map +1 -0
  30. package/dist-server/index.d.ts +1 -0
  31. package/dist-server/index.js +2 -0
  32. package/dist-server/index.js.map +1 -1
  33. package/dist-server/routers/pdf-private-router.d.ts +1 -0
  34. package/dist-server/routers/pdf-private-router.js +72 -0
  35. package/dist-server/routers/pdf-private-router.js.map +1 -0
  36. package/dist-server/routers/pdf-public-router.d.ts +1 -0
  37. package/dist-server/routers/{pdf-router.js → pdf-public-router.js} +6 -6
  38. package/dist-server/routers/pdf-public-router.js.map +1 -0
  39. package/dist-server/routes.js +13 -2
  40. package/dist-server/routes.js.map +1 -1
  41. package/dist-server/service/index.d.ts +7 -0
  42. package/dist-server/service/index.js +31 -0
  43. package/dist-server/service/index.js.map +1 -0
  44. package/dist-server/service/pdf-generate/pdf-generate-resolver.d.ts +6 -0
  45. package/dist-server/service/pdf-generate/pdf-generate-resolver.js +88 -0
  46. package/dist-server/service/pdf-generate/pdf-generate-resolver.js.map +1 -0
  47. package/dist-server/service/pdf-release/index.d.ts +6 -0
  48. package/dist-server/service/pdf-release/index.js +10 -0
  49. package/dist-server/service/pdf-release/index.js.map +1 -0
  50. package/dist-server/service/pdf-release/pdf-release-mutation.d.ts +10 -0
  51. package/dist-server/service/pdf-release/pdf-release-mutation.js +127 -0
  52. package/dist-server/service/pdf-release/pdf-release-mutation.js.map +1 -0
  53. package/dist-server/service/pdf-release/pdf-release-query.d.ts +11 -0
  54. package/dist-server/service/pdf-release/pdf-release-query.js +79 -0
  55. package/dist-server/service/pdf-release/pdf-release-query.js.map +1 -0
  56. package/dist-server/service/pdf-release/pdf-release-type.d.ts +20 -0
  57. package/dist-server/service/pdf-release/pdf-release-type.js +77 -0
  58. package/dist-server/service/pdf-release/pdf-release-type.js.map +1 -0
  59. package/dist-server/service/pdf-release/pdf-release.d.ts +27 -0
  60. package/dist-server/service/pdf-release/pdf-release.js +117 -0
  61. package/dist-server/service/pdf-release/pdf-release.js.map +1 -0
  62. package/dist-server/service/pdf-template/index.d.ts +6 -0
  63. package/dist-server/service/pdf-template/index.js +10 -0
  64. package/dist-server/service/pdf-template/index.js.map +1 -0
  65. package/dist-server/service/pdf-template/pdf-template-mutation.d.ts +10 -0
  66. package/dist-server/service/pdf-template/pdf-template-mutation.js +127 -0
  67. package/dist-server/service/pdf-template/pdf-template-mutation.js.map +1 -0
  68. package/dist-server/service/pdf-template/pdf-template-query.d.ts +11 -0
  69. package/dist-server/service/pdf-template/pdf-template-query.js +79 -0
  70. package/dist-server/service/pdf-template/pdf-template-query.js.map +1 -0
  71. package/dist-server/service/pdf-template/pdf-template-type.d.ts +32 -0
  72. package/dist-server/service/pdf-template/pdf-template-type.js +125 -0
  73. package/dist-server/service/pdf-template/pdf-template-type.js.map +1 -0
  74. package/dist-server/service/pdf-template/pdf-template.d.ts +28 -0
  75. package/dist-server/service/pdf-template/pdf-template.js +125 -0
  76. package/dist-server/service/pdf-template/pdf-template.js.map +1 -0
  77. package/dist-server/tsconfig.tsbuildinfo +1 -1
  78. package/helps/pdf/pdf-release.md +160 -0
  79. package/helps/pdf/pdf-template.md +160 -0
  80. package/package.json +15 -8
  81. package/server/controller/pdf-service.ts +35 -0
  82. package/server/index.ts +2 -0
  83. package/server/routers/pdf-private-router.ts +85 -0
  84. package/server/routers/{pdf-router.ts → pdf-public-router.ts} +4 -4
  85. package/server/routes.ts +14 -2
  86. package/server/service/index.ts +38 -0
  87. package/server/service/pdf-generate/pdf-generate-resolver.ts +81 -0
  88. package/server/service/pdf-release/index.ts +7 -0
  89. package/server/service/pdf-release/pdf-release-mutation.ts +138 -0
  90. package/server/service/pdf-release/pdf-release-query.ts +51 -0
  91. package/server/service/pdf-release/pdf-release-type.ts +55 -0
  92. package/server/service/pdf-release/pdf-release.ts +103 -0
  93. package/server/service/pdf-template/index.ts +7 -0
  94. package/server/service/pdf-template/pdf-template-mutation.ts +138 -0
  95. package/server/service/pdf-template/pdf-template-query.ts +51 -0
  96. package/server/service/pdf-template/pdf-template-type.ts +87 -0
  97. package/server/service/pdf-template/pdf-template.ts +108 -0
  98. package/server/tsconfig.json +10 -0
  99. package/things-factory.config.js +8 -1
  100. package/translations/en.json +11 -0
  101. package/translations/ja.json +9 -0
  102. package/translations/ko.json +11 -0
  103. package/translations/ms.json +9 -0
  104. package/translations/zh.json +9 -0
  105. package/dist-server/routers/pdf-router.d.ts +0 -1
  106. package/dist-server/routers/pdf-router.js.map +0 -1
  107. package/tsconfig.json +0 -9
@@ -0,0 +1,160 @@
1
+ # PdfRelease
2
+
3
+ Paragraphs are separated by a blank line.
4
+
5
+ 2nd paragraph. _Italic_, **bold**, and `monospace`. Itemized lists
6
+ look like:
7
+
8
+ - this one
9
+ - that one
10
+ - the other one
11
+
12
+ Note that --- not considering the asterisk --- the actual text
13
+ content starts at 4-columns in.
14
+
15
+ > Block quotes are
16
+ > written like so.
17
+ >
18
+ > They can span multiple paragraphs,
19
+ > if you like.
20
+
21
+ Use 3 dashes for an em-dash. Use 2 dashes for ranges (ex., "it's all
22
+ in chapters 12--14"). Three dots ... will be converted to an ellipsis.
23
+ Unicode is supported. ☺
24
+
25
+ ## An h2 header
26
+
27
+ Here's a numbered list:
28
+
29
+ 1. first item
30
+ 2. second item
31
+ 3. third item
32
+
33
+ Note again how the actual text starts at 4 columns in (4 characters
34
+ from the left side). Here's a code sample:
35
+
36
+ # Let me re-iterate ...
37
+
38
+ for i in 1 .. 10 { do-something(i) }
39
+
40
+ As you probably guessed, indented 4 spaces. By the way, instead of
41
+ indenting the block, you can use delimited blocks, if you like:
42
+
43
+ ```
44
+ define foobar() {
45
+ print "Welcome to flavor country!";
46
+ }
47
+ ```
48
+
49
+ (which makes copying & pasting easier). You can optionally mark the
50
+ delimited block for Pandoc to syntax highlight it:
51
+
52
+ ```python
53
+ import time
54
+ # Quick, count to ten!
55
+ for i in range(10):
56
+ # (but not *too* quick)
57
+ time.sleep(0.5)
58
+ print(i)
59
+ ```
60
+
61
+ ### An h3 header
62
+
63
+ Now a nested list:
64
+
65
+ 1. First, get these ingredients:
66
+
67
+ - carrots
68
+ - celery
69
+ - lentils
70
+
71
+ 2. Boil some water.
72
+
73
+ 3. Dump everything in the pot and follow
74
+ this algorithm:
75
+
76
+ find wooden spoon
77
+ uncover pot
78
+ stir
79
+ cover pot
80
+ balance wooden spoon precariously on pot handle
81
+ wait 10 minutes
82
+ goto first step (or shut off burner when done)
83
+
84
+ Do not bump wooden spoon or it will fall.
85
+
86
+ Notice again how text always lines up on 4-space indents (including
87
+ that last line which continues item 3 above).
88
+
89
+ Here's a link to [a website](http://foo.bar), to a [local
90
+ doc](local-doc.html), and to a [section heading in the current
91
+ doc](#an-h2-header). Here's a footnote [^1].
92
+
93
+ [^1]: Some footnote text.
94
+
95
+ Tables can look like this:
96
+
97
+ Name Size Material Color
98
+
99
+ ---
100
+
101
+ All Business 9 leather brown
102
+ Roundabout 10 hemp canvas natural
103
+ Cinderella 11 glass transparent
104
+
105
+ Table: Shoes sizes, materials, and colors.
106
+
107
+ (The above is the caption for the table.) Pandoc also supports
108
+ multi-line tables:
109
+
110
+ ---
111
+
112
+ Keyword Text
113
+
114
+ ---
115
+
116
+ red Sunsets, apples, and
117
+ other red or reddish
118
+ things.
119
+
120
+ green Leaves, grass, frogs
121
+ and other things it's
122
+ not easy being.
123
+
124
+ ---
125
+
126
+ A horizontal rule follows.
127
+
128
+ ---
129
+
130
+ Here's a definition list:
131
+
132
+ apples
133
+ : Good for making applesauce.
134
+
135
+ oranges
136
+ : Citrus!
137
+
138
+ tomatoes
139
+ : There's no "e" in tomatoe.
140
+
141
+ Again, text is indented 4 spaces. (Put a blank line between each
142
+ term and its definition to spread things out more.)
143
+
144
+ Here's a "line block" (note how whitespace is honored):
145
+
146
+ | Line one
147
+ | Line too
148
+ | Line tree
149
+
150
+ and images can be specified like so:
151
+
152
+ ![example image](example-image.jpg 'An exemplary image')
153
+
154
+ Inline math equation: $\omega = d\phi / dt$. Display
155
+ math should get its own line like so:
156
+
157
+ $$I = \int \rho R^{2} dV$$
158
+
159
+ And note that you can backslash-escape any punctuation characters
160
+ which you wish to be displayed literally, ex.: \`foo\`, \*bar\*, etc.
@@ -0,0 +1,160 @@
1
+ # PdfTemplate
2
+
3
+ Paragraphs are separated by a blank line.
4
+
5
+ 2nd paragraph. _Italic_, **bold**, and `monospace`. Itemized lists
6
+ look like:
7
+
8
+ - this one
9
+ - that one
10
+ - the other one
11
+
12
+ Note that --- not considering the asterisk --- the actual text
13
+ content starts at 4-columns in.
14
+
15
+ > Block quotes are
16
+ > written like so.
17
+ >
18
+ > They can span multiple paragraphs,
19
+ > if you like.
20
+
21
+ Use 3 dashes for an em-dash. Use 2 dashes for ranges (ex., "it's all
22
+ in chapters 12--14"). Three dots ... will be converted to an ellipsis.
23
+ Unicode is supported. ☺
24
+
25
+ ## An h2 header
26
+
27
+ Here's a numbered list:
28
+
29
+ 1. first item
30
+ 2. second item
31
+ 3. third item
32
+
33
+ Note again how the actual text starts at 4 columns in (4 characters
34
+ from the left side). Here's a code sample:
35
+
36
+ # Let me re-iterate ...
37
+
38
+ for i in 1 .. 10 { do-something(i) }
39
+
40
+ As you probably guessed, indented 4 spaces. By the way, instead of
41
+ indenting the block, you can use delimited blocks, if you like:
42
+
43
+ ```
44
+ define foobar() {
45
+ print "Welcome to flavor country!";
46
+ }
47
+ ```
48
+
49
+ (which makes copying & pasting easier). You can optionally mark the
50
+ delimited block for Pandoc to syntax highlight it:
51
+
52
+ ```python
53
+ import time
54
+ # Quick, count to ten!
55
+ for i in range(10):
56
+ # (but not *too* quick)
57
+ time.sleep(0.5)
58
+ print(i)
59
+ ```
60
+
61
+ ### An h3 header
62
+
63
+ Now a nested list:
64
+
65
+ 1. First, get these ingredients:
66
+
67
+ - carrots
68
+ - celery
69
+ - lentils
70
+
71
+ 2. Boil some water.
72
+
73
+ 3. Dump everything in the pot and follow
74
+ this algorithm:
75
+
76
+ find wooden spoon
77
+ uncover pot
78
+ stir
79
+ cover pot
80
+ balance wooden spoon precariously on pot handle
81
+ wait 10 minutes
82
+ goto first step (or shut off burner when done)
83
+
84
+ Do not bump wooden spoon or it will fall.
85
+
86
+ Notice again how text always lines up on 4-space indents (including
87
+ that last line which continues item 3 above).
88
+
89
+ Here's a link to [a website](http://foo.bar), to a [local
90
+ doc](local-doc.html), and to a [section heading in the current
91
+ doc](#an-h2-header). Here's a footnote [^1].
92
+
93
+ [^1]: Some footnote text.
94
+
95
+ Tables can look like this:
96
+
97
+ Name Size Material Color
98
+
99
+ ---
100
+
101
+ All Business 9 leather brown
102
+ Roundabout 10 hemp canvas natural
103
+ Cinderella 11 glass transparent
104
+
105
+ Table: Shoes sizes, materials, and colors.
106
+
107
+ (The above is the caption for the table.) Pandoc also supports
108
+ multi-line tables:
109
+
110
+ ---
111
+
112
+ Keyword Text
113
+
114
+ ---
115
+
116
+ red Sunsets, apples, and
117
+ other red or reddish
118
+ things.
119
+
120
+ green Leaves, grass, frogs
121
+ and other things it's
122
+ not easy being.
123
+
124
+ ---
125
+
126
+ A horizontal rule follows.
127
+
128
+ ---
129
+
130
+ Here's a definition list:
131
+
132
+ apples
133
+ : Good for making applesauce.
134
+
135
+ oranges
136
+ : Citrus!
137
+
138
+ tomatoes
139
+ : There's no "e" in tomatoe.
140
+
141
+ Again, text is indented 4 spaces. (Put a blank line between each
142
+ term and its definition to spread things out more.)
143
+
144
+ Here's a "line block" (note how whitespace is honored):
145
+
146
+ | Line one
147
+ | Line too
148
+ | Line tree
149
+
150
+ and images can be specified like so:
151
+
152
+ ![example image](example-image.jpg 'An exemplary image')
153
+
154
+ Inline math equation: $\omega = d\phi / dt$. Display
155
+ math should get its own line like so:
156
+
157
+ $$I = \int \rho R^{2} dV$$
158
+
159
+ And note that you can backslash-escape any punctuation characters
160
+ which you wish to be displayed literally, ex.: \`foo\`, \*bar\*, etc.
package/package.json CHANGED
@@ -1,7 +1,8 @@
1
1
  {
2
2
  "name": "@things-factory/pdf",
3
- "version": "7.0.33",
3
+ "version": "7.0.37",
4
4
  "main": "dist-server/index.js",
5
+ "browser": "dist-client/index.js",
5
6
  "things-factory": true,
6
7
  "author": "heartyoh <heartyoh@hatiolab.com>",
7
8
  "description": "Module that provide services for merging pdf files and extracting image from pdf.",
@@ -16,15 +17,21 @@
16
17
  "directory": "packages/pdf"
17
18
  },
18
19
  "scripts": {
19
- "build": "tsc --p tsconfig.json",
20
- "build:server": "npm run clean:server && tsc",
21
- "clean:server": "rm -rf dist-server",
22
- "clean": "npm run clean:server",
23
- "migration:create": "node ../../node_modules/typeorm/cli.js migration:create -d ./server/migrations"
20
+ "build": "npm run build:server && npm run build:client",
21
+ "copy:files": "copyfiles -e \"./client/**/*.{ts,js,json}\" -u 1 \"./client/**/*\" dist-client",
22
+ "build:client": "npm run copy:files && npx tsc --p ./client/tsconfig.json",
23
+ "build:server": "npx tsc --p ./server/tsconfig.json",
24
+ "clean:client": "npx rimraf dist-client",
25
+ "clean:server": "npx rimraf dist-server",
26
+ "clean": "npm run clean:server && npm run clean:client"
24
27
  },
25
28
  "dependencies": {
29
+ "@things-factory/auth-base": "^7.0.33",
26
30
  "@things-factory/utils": "^7.0.33",
27
- "await-spawn": "^4.0.1"
31
+ "await-spawn": "^4.0.1",
32
+ "ejs": "^3.1.10",
33
+ "handlebars": "^4.7.8",
34
+ "pdf-lib": "^1.17.1"
28
35
  },
29
- "gitHead": "d8ce7eb155cf54e1c8aa9dca34ca7872b0a0c804"
36
+ "gitHead": "28b852bbf7bd1160522195f9db1775db56f18b30"
30
37
  }
@@ -0,0 +1,35 @@
1
+ import * as ejs from 'ejs'
2
+ import { PDFDocument } from 'pdf-lib'
3
+ import { PDFTemplate } from '../service/pdf-template/pdf-template'
4
+
5
+ export class PdfService {
6
+ async generatePdf(template: PDFTemplate, data: any): Promise<Buffer> {
7
+ const pdfDoc = await PDFDocument.create()
8
+
9
+ // 템플릿 렌더링
10
+ const renderedContent = await ejs.render(template.content_template || '', data)
11
+
12
+ // PDF 생성
13
+ const page = pdfDoc.addPage()
14
+ const { width, height } = page.getSize()
15
+ page.drawText(renderedContent, {
16
+ x: 50,
17
+ y: height - 50,
18
+ size: 12
19
+ })
20
+
21
+ // 추가적으로 header, footer, cover 등의 렌더링 및 PDF 구성 가능
22
+ if (template.header_template) {
23
+ const renderedHeader = await ejs.render(template.header_template, data)
24
+ // Header 처리 로직
25
+ }
26
+
27
+ if (template.footer_template) {
28
+ const renderedFooter = await ejs.render(template.footer_template, data)
29
+ // Footer 처리 로직
30
+ }
31
+
32
+ const pdfBytes = await pdfDoc.save()
33
+ return Buffer.from(pdfBytes)
34
+ }
35
+ }
package/server/index.ts CHANGED
@@ -1 +1,3 @@
1
+ export * from './service'
2
+
1
3
  import './routes'
@@ -0,0 +1,85 @@
1
+ import Router from 'koa-router'
2
+ import * as fs from 'fs'
3
+ import * as path from 'path'
4
+ import send from 'koa-send'
5
+
6
+ import { PDFRelease } from '../service/pdf-release/pdf-release'
7
+ import { PDFTemplate } from '../service/pdf-template/pdf-template'
8
+ import { PdfService } from '../controller/pdf-service'
9
+
10
+ const PDF_STORAGE_PATH = '/path/to/pdf/storage' // 실제 파일 저장 경로
11
+ const PDF_BASE_URL = 'https://yourdomain.com/pdf/' // 파일 접근 URL의 기본 경로
12
+
13
+ const pdfService = new PdfService()
14
+
15
+ export const pdfPrivateRouter = new Router()
16
+
17
+ function parseQuery(query) {
18
+ for (const key in query) {
19
+ if (query.hasOwnProperty(key)) {
20
+ try {
21
+ query[key] = JSON.parse(query[key])
22
+ } catch (error) {
23
+ // do nothing
24
+ }
25
+ }
26
+ }
27
+
28
+ return query
29
+ }
30
+
31
+ // PDF 생성 및 저장 라우터
32
+ pdfPrivateRouter.post('/generate-pdf', async ctx => {
33
+ const { templateId, data } = ctx.request.body
34
+ const { domain, user, tx } = ctx.state
35
+
36
+ const templateRepository = tx.getRepository(PDFTemplate)
37
+ const releaseRepository = tx.getRepository(PDFRelease)
38
+
39
+ const template = await templateRepository.findOne(templateId)
40
+ if (!template) {
41
+ ctx.throw(404, 'PDF Template not found')
42
+ }
43
+
44
+ // EJS 템플릿을 사용하여 PDF 생성
45
+ const pdfBuffer = await pdfService.generatePdf(template, data)
46
+
47
+ // 파일 저장 경로 및 URL 생성
48
+ const fileName = `${template.name}-${Date.now()}.pdf`
49
+ const filePath = path.join(PDF_STORAGE_PATH, fileName)
50
+ const fileUrl = `${PDF_BASE_URL}${fileName}`
51
+
52
+ // 파일 저장
53
+ fs.writeFileSync(filePath, pdfBuffer)
54
+
55
+ // 발행 이력 기록
56
+ const release = releaseRepository.create({
57
+ template,
58
+ filePath,
59
+ fileUrl,
60
+ state: 'published',
61
+ releasedBy: user,
62
+ domain
63
+ })
64
+
65
+ await releaseRepository.save(release)
66
+
67
+ ctx.body = release
68
+ })
69
+
70
+ // PDF 다운로드 라우터
71
+ pdfPrivateRouter.get('/download-pdf/:id', async ctx => {
72
+ const { id } = ctx.params
73
+ const { tx } = ctx.state
74
+
75
+ const releaseRepository = tx.getRepository(PDFRelease)
76
+ const pdfRelease = await releaseRepository.findOne(id)
77
+
78
+ if (!pdfRelease || !pdfRelease.filePath || !fs.existsSync(pdfRelease.filePath)) {
79
+ ctx.throw(404, 'PDF file not found')
80
+ }
81
+
82
+ // PDF 파일을 클라이언트에 전송
83
+ ctx.attachment(path.basename(pdfRelease.filePath))
84
+ await send(ctx, pdfRelease.filePath, { root: '/' })
85
+ })
@@ -4,10 +4,10 @@ import Router from 'koa-router'
4
4
  import path from 'path'
5
5
  import { withTempDir } from '@things-factory/utils'
6
6
 
7
- export const pdfRouter = new Router()
7
+ export const pdfPublicRouter = new Router()
8
8
 
9
9
  /* to-png */
10
- pdfRouter.get('/to-png', async (context, next) => {
10
+ pdfPublicRouter.get('/to-png', async (context, next) => {
11
11
  const { url } = context.query
12
12
 
13
13
  const images = await withTempDir(async tmpDir => {
@@ -60,7 +60,7 @@ const mergePDFfiles = async urls => {
60
60
  })
61
61
  }
62
62
 
63
- pdfRouter.get('/merge', async (context, next) => {
63
+ pdfPublicRouter.get('/merge', async (context, next) => {
64
64
  const searchParams = new URLSearchParams(context.querystring)
65
65
  const urls = searchParams.getAll('urls')
66
66
 
@@ -70,7 +70,7 @@ pdfRouter.get('/merge', async (context, next) => {
70
70
  context.body = pdf
71
71
  })
72
72
 
73
- pdfRouter.post('/merge', async (context, next) => {
73
+ pdfPublicRouter.post('/merge', async (context, next) => {
74
74
  const { urls } = context.request.body
75
75
 
76
76
  const pdf = await mergePDFfiles(urls)
package/server/routes.ts CHANGED
@@ -1,7 +1,19 @@
1
- import { pdfRouter } from './routers/pdf-router'
1
+ import { config } from '@things-factory/env'
2
+ import { pdfPublicRouter } from './routers/pdf-public-router'
3
+ import { pdfPrivateRouter } from './routers/pdf-private-router'
2
4
  import { proxyRouter } from './routers/proxy-router'
3
5
 
6
+ const isPathBaseDomain = !config.get('subdomain') && !config.get('useVirtualHostBasedDomain')
7
+
4
8
  process.on('bootstrap-module-global-public-route' as any, (app, globalPublicRouter) => {
5
- globalPublicRouter.use('/pdf', pdfRouter.routes(), pdfRouter.allowedMethods())
9
+ globalPublicRouter.use('/pdf', pdfPublicRouter.routes(), pdfPublicRouter.allowedMethods())
6
10
  globalPublicRouter.use('/proxy', proxyRouter.routes(), proxyRouter.allowedMethods())
7
11
  })
12
+
13
+ process.on('bootstrap-module-domain-private-route' as any, (app, domainPrivateRouter) => {
14
+ if (isPathBaseDomain) {
15
+ domainPrivateRouter.use('/domain/:domain', pdfPrivateRouter.routes(), pdfPrivateRouter.allowedMethods())
16
+ } else {
17
+ domainPrivateRouter.use('', pdfPrivateRouter.routes(), pdfPrivateRouter.allowedMethods())
18
+ }
19
+ })
@@ -0,0 +1,38 @@
1
+ /* EXPORT ENTITY TYPES */
2
+ export * from './pdf-release/pdf-release'
3
+ export * from './pdf-template/pdf-template'
4
+
5
+ /* IMPORT ENTITIES AND RESOLVERS */
6
+ import {
7
+ entities as PdfReleaseEntities,
8
+ resolvers as PdfReleaseResolvers,
9
+ subscribers as PdfReleaseSubscribers
10
+ } from './pdf-release'
11
+ import {
12
+ entities as PDFTemplateEntities,
13
+ resolvers as PDFTemplateResolvers,
14
+ subscribers as PDFTemplateSubscribers
15
+ } from './pdf-template'
16
+
17
+ export const entities = [
18
+ /* ENTITIES */
19
+ ...PdfReleaseEntities,
20
+ ...PdfReleaseEntities,
21
+ ...PDFTemplateEntities
22
+ ]
23
+
24
+ export const subscribers = [
25
+ /* SUBSCRIBERS */
26
+ ...PdfReleaseSubscribers,
27
+ ...PdfReleaseSubscribers,
28
+ ...PDFTemplateSubscribers
29
+ ]
30
+
31
+ export const schema = {
32
+ resolverClasses: [
33
+ /* RESOLVER CLASSES */
34
+ ...PdfReleaseResolvers,
35
+ ...PdfReleaseResolvers,
36
+ ...PDFTemplateResolvers
37
+ ]
38
+ }
@@ -0,0 +1,81 @@
1
+ import { Resolver, Mutation, Arg, Ctx, Query } from 'type-graphql'
2
+ import * as fs from 'fs'
3
+ import * as path from 'path'
4
+
5
+ import { PDFRelease, PDFReleaseStatus } from '../pdf-release/pdf-release'
6
+ import { PDFTemplate } from '../pdf-template/pdf-template'
7
+ import { PdfService } from '../../controller/pdf-service'
8
+ import send from 'koa-send'
9
+
10
+ const PDF_STORAGE_PATH = '/path/to/pdf/storage' // 실제 파일 저장 경로
11
+ const PDF_BASE_URL = 'https://yourdomain.com/pdf/' // 파일 접근 URL의 기본 경로
12
+
13
+ @Resolver()
14
+ export class PDFGenerateResolver {
15
+ private pdfService = new PdfService()
16
+
17
+ @Mutation(() => PDFRelease)
18
+ async generatePDF(
19
+ @Arg('templateId') templateId: string,
20
+ @Arg('data') data: any,
21
+ @Ctx() ctx: ResolverContext
22
+ ): Promise<PDFRelease> {
23
+ const { domain, user, tx } = ctx.state
24
+ const templateRepository = tx.getRepository(PDFTemplate)
25
+ const releaseRepository = tx.getRepository(PDFRelease)
26
+
27
+ const template = await templateRepository.findOne({ where: { id: templateId, domain: { id: domain.id } } })
28
+ if (!template) {
29
+ throw new Error('PDF Template not found')
30
+ }
31
+
32
+ var state: PDFReleaseStatus = PDFReleaseStatus.published
33
+
34
+ try {
35
+ const pdfBuffer = await this.pdfService.generatePdf(template, data)
36
+
37
+ const fileName = `${template.name}-${Date.now()}.pdf`
38
+ const filePath = path.join(PDF_STORAGE_PATH, fileName)
39
+ const fileUrl = `${PDF_BASE_URL}${fileName}`
40
+
41
+ fs.writeFileSync(filePath, pdfBuffer)
42
+
43
+ return releaseRepository.save({
44
+ template,
45
+ filePath,
46
+ fileUrl,
47
+ state,
48
+ creator: user,
49
+ updater: user,
50
+ domain
51
+ })
52
+ } catch (error) {
53
+ state = PDFReleaseStatus.failed
54
+ return releaseRepository.save({
55
+ template,
56
+ state,
57
+ creator: user,
58
+ updater: user,
59
+ domain
60
+ })
61
+ }
62
+ }
63
+
64
+ @Query(() => String, { description: 'Download the generated PDF file' })
65
+ async downloadPDF(@Arg('id') id: string, @Ctx() ctx: ResolverContext): Promise<string> {
66
+ const { domain, user, tx } = ctx.state
67
+
68
+ const releaseRepository = tx.getRepository(PDFRelease)
69
+ const pdfRelease = await releaseRepository.findOne({ where: { id, domain: { id: domain.id } } })
70
+
71
+ if (!pdfRelease || !pdfRelease.filePath || !fs.existsSync(pdfRelease.filePath)) {
72
+ throw new Error('PDF file not found')
73
+ }
74
+
75
+ // PDF 파일을 클라이언트에 전송
76
+ ctx.attachment(path.basename(pdfRelease.filePath))
77
+ await send(ctx, pdfRelease.filePath, { root: '/' })
78
+
79
+ return 'Download started'
80
+ }
81
+ }
@@ -0,0 +1,7 @@
1
+ import { PDFRelease } from './pdf-release'
2
+ import { PDFReleaseQuery } from './pdf-release-query'
3
+ import { PDFReleaseMutation } from './pdf-release-mutation'
4
+
5
+ export const entities = [PDFRelease]
6
+ export const resolvers = [PDFReleaseQuery, PDFReleaseMutation]
7
+ export const subscribers = []