@things-factory/pdf 7.0.36 → 7.0.43
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.
- package/client/pages/pdf-release/pdf-release-importer.ts +90 -0
- package/client/pages/pdf-release/pdf-release-list-page.ts +398 -0
- package/client/route.ts +5 -1
- package/dist-client/pages/pdf-release/pdf-release-importer.d.ts +23 -0
- package/dist-client/pages/pdf-release/pdf-release-importer.js +93 -0
- package/dist-client/pages/pdf-release/pdf-release-importer.js.map +1 -0
- package/dist-client/pages/pdf-release/pdf-release-list-page.d.ts +66 -0
- package/dist-client/pages/pdf-release/pdf-release-list-page.js +370 -0
- package/dist-client/pages/pdf-release/pdf-release-list-page.js.map +1 -0
- package/dist-client/route.d.ts +1 -1
- package/dist-client/route.js +3 -0
- package/dist-client/route.js.map +1 -1
- package/dist-client/tsconfig.tsbuildinfo +1 -1
- package/dist-server/controller/pdf-service.d.ts +4 -0
- package/dist-server/controller/pdf-service.js +34 -0
- package/dist-server/controller/pdf-service.js.map +1 -0
- package/dist-server/routers/pdf-private-router.js +51 -4
- package/dist-server/routers/pdf-private-router.js.map +1 -1
- package/dist-server/service/index.d.ts +3 -2
- package/dist-server/service/index.js +8 -0
- package/dist-server/service/index.js.map +1 -1
- package/dist-server/service/pdf-generate/pdf-generate-resolver.d.ts +6 -0
- package/dist-server/service/pdf-generate/pdf-generate-resolver.js +88 -0
- package/dist-server/service/pdf-generate/pdf-generate-resolver.js.map +1 -0
- package/dist-server/service/pdf-release/index.d.ts +6 -0
- package/dist-server/service/pdf-release/index.js +10 -0
- package/dist-server/service/pdf-release/index.js.map +1 -0
- package/dist-server/service/pdf-release/pdf-release-mutation.d.ts +10 -0
- package/dist-server/service/pdf-release/pdf-release-mutation.js +127 -0
- package/dist-server/service/pdf-release/pdf-release-mutation.js.map +1 -0
- package/dist-server/service/pdf-release/pdf-release-query.d.ts +11 -0
- package/dist-server/service/pdf-release/pdf-release-query.js +79 -0
- package/dist-server/service/pdf-release/pdf-release-query.js.map +1 -0
- package/dist-server/service/pdf-release/pdf-release-type.d.ts +20 -0
- package/dist-server/service/pdf-release/pdf-release-type.js +77 -0
- package/dist-server/service/pdf-release/pdf-release-type.js.map +1 -0
- package/dist-server/service/pdf-release/pdf-release.d.ts +27 -0
- package/dist-server/service/pdf-release/pdf-release.js +117 -0
- package/dist-server/service/pdf-release/pdf-release.js.map +1 -0
- package/dist-server/tsconfig.tsbuildinfo +1 -1
- package/helps/pdf/pdf-release.md +160 -0
- package/package.json +6 -4
- package/server/controller/pdf-service.ts +35 -0
- package/server/routers/pdf-private-router.ts +65 -4
- package/server/service/index.ts +12 -0
- package/server/service/pdf-generate/pdf-generate-resolver.ts +81 -0
- package/server/service/pdf-release/index.ts +7 -0
- package/server/service/pdf-release/pdf-release-mutation.ts +138 -0
- package/server/service/pdf-release/pdf-release-query.ts +51 -0
- package/server/service/pdf-release/pdf-release-type.ts +55 -0
- package/server/service/pdf-release/pdf-release.ts +103 -0
|
@@ -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
|
+

|
|
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,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@things-factory/pdf",
|
|
3
|
-
"version": "7.0.
|
|
3
|
+
"version": "7.0.43",
|
|
4
4
|
"main": "dist-server/index.js",
|
|
5
5
|
"browser": "dist-client/index.js",
|
|
6
6
|
"things-factory": true,
|
|
@@ -26,10 +26,12 @@
|
|
|
26
26
|
"clean": "npm run clean:server && npm run clean:client"
|
|
27
27
|
},
|
|
28
28
|
"dependencies": {
|
|
29
|
-
"@things-factory/auth-base": "^7.0.
|
|
29
|
+
"@things-factory/auth-base": "^7.0.43",
|
|
30
30
|
"@things-factory/utils": "^7.0.33",
|
|
31
31
|
"await-spawn": "^4.0.1",
|
|
32
|
-
"
|
|
32
|
+
"ejs": "^3.1.10",
|
|
33
|
+
"handlebars": "^4.7.8",
|
|
34
|
+
"pdf-lib": "^1.17.1"
|
|
33
35
|
},
|
|
34
|
-
"gitHead": "
|
|
36
|
+
"gitHead": "669cf13dccf67bd45260d8f69d919bc1d72c1ce0"
|
|
35
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
|
+
}
|
|
@@ -1,4 +1,16 @@
|
|
|
1
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()
|
|
2
14
|
|
|
3
15
|
export const pdfPrivateRouter = new Router()
|
|
4
16
|
|
|
@@ -16,9 +28,58 @@ function parseQuery(query) {
|
|
|
16
28
|
return query
|
|
17
29
|
}
|
|
18
30
|
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
const data =
|
|
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
|
+
}
|
|
22
81
|
|
|
23
|
-
|
|
82
|
+
// PDF 파일을 클라이언트에 전송
|
|
83
|
+
ctx.attachment(path.basename(pdfRelease.filePath))
|
|
84
|
+
await send(ctx, pdfRelease.filePath, { root: '/' })
|
|
24
85
|
})
|
package/server/service/index.ts
CHANGED
|
@@ -1,7 +1,13 @@
|
|
|
1
1
|
/* EXPORT ENTITY TYPES */
|
|
2
|
+
export * from './pdf-release/pdf-release'
|
|
2
3
|
export * from './pdf-template/pdf-template'
|
|
3
4
|
|
|
4
5
|
/* IMPORT ENTITIES AND RESOLVERS */
|
|
6
|
+
import {
|
|
7
|
+
entities as PdfReleaseEntities,
|
|
8
|
+
resolvers as PdfReleaseResolvers,
|
|
9
|
+
subscribers as PdfReleaseSubscribers
|
|
10
|
+
} from './pdf-release'
|
|
5
11
|
import {
|
|
6
12
|
entities as PDFTemplateEntities,
|
|
7
13
|
resolvers as PDFTemplateResolvers,
|
|
@@ -10,17 +16,23 @@ import {
|
|
|
10
16
|
|
|
11
17
|
export const entities = [
|
|
12
18
|
/* ENTITIES */
|
|
19
|
+
...PdfReleaseEntities,
|
|
20
|
+
...PdfReleaseEntities,
|
|
13
21
|
...PDFTemplateEntities
|
|
14
22
|
]
|
|
15
23
|
|
|
16
24
|
export const subscribers = [
|
|
17
25
|
/* SUBSCRIBERS */
|
|
26
|
+
...PdfReleaseSubscribers,
|
|
27
|
+
...PdfReleaseSubscribers,
|
|
18
28
|
...PDFTemplateSubscribers
|
|
19
29
|
]
|
|
20
30
|
|
|
21
31
|
export const schema = {
|
|
22
32
|
resolverClasses: [
|
|
23
33
|
/* RESOLVER CLASSES */
|
|
34
|
+
...PdfReleaseResolvers,
|
|
35
|
+
...PdfReleaseResolvers,
|
|
24
36
|
...PDFTemplateResolvers
|
|
25
37
|
]
|
|
26
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 = []
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import { Resolver, Mutation, Arg, Ctx, Directive } from 'type-graphql'
|
|
2
|
+
import { In } from 'typeorm'
|
|
3
|
+
|
|
4
|
+
import { PDFRelease } from './pdf-release'
|
|
5
|
+
import { NewPDFRelease, PDFReleasePatch } from './pdf-release-type'
|
|
6
|
+
|
|
7
|
+
@Resolver(PDFRelease)
|
|
8
|
+
export class PDFReleaseMutation {
|
|
9
|
+
@Directive('@transaction')
|
|
10
|
+
@Mutation(returns => PDFRelease, { description: 'To create new PDFRelease' })
|
|
11
|
+
async createPDFRelease(
|
|
12
|
+
@Arg('pdfRelease') pdfRelease: NewPDFRelease,
|
|
13
|
+
@Ctx() context: ResolverContext
|
|
14
|
+
): Promise<PDFRelease> {
|
|
15
|
+
const { domain, user, tx } = context.state
|
|
16
|
+
|
|
17
|
+
const result = await tx.getRepository(PDFRelease).save({
|
|
18
|
+
...pdfRelease,
|
|
19
|
+
domain,
|
|
20
|
+
creator: user,
|
|
21
|
+
updater: user
|
|
22
|
+
})
|
|
23
|
+
|
|
24
|
+
return result
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
@Directive('@transaction')
|
|
28
|
+
@Mutation(returns => PDFRelease, { description: 'To modify PDFRelease information' })
|
|
29
|
+
async updatePDFRelease(
|
|
30
|
+
@Arg('id') id: string,
|
|
31
|
+
@Arg('patch') patch: PDFReleasePatch,
|
|
32
|
+
@Ctx() context: ResolverContext
|
|
33
|
+
): Promise<PDFRelease> {
|
|
34
|
+
const { domain, user, tx } = context.state
|
|
35
|
+
|
|
36
|
+
const repository = tx.getRepository(PDFRelease)
|
|
37
|
+
const pdfRelease = await repository.findOne({
|
|
38
|
+
where: { domain: { id: domain.id }, id }
|
|
39
|
+
})
|
|
40
|
+
|
|
41
|
+
const result = await repository.save({
|
|
42
|
+
...pdfRelease,
|
|
43
|
+
...patch,
|
|
44
|
+
updater: user
|
|
45
|
+
})
|
|
46
|
+
|
|
47
|
+
return result
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
@Directive('@transaction')
|
|
51
|
+
@Mutation(returns => [PDFRelease], { description: "To modify multiple PDFReleases' information" })
|
|
52
|
+
async updateMultiplePDFRelease(
|
|
53
|
+
@Arg('patches', type => [PDFReleasePatch]) patches: PDFReleasePatch[],
|
|
54
|
+
@Ctx() context: ResolverContext
|
|
55
|
+
): Promise<PDFRelease[]> {
|
|
56
|
+
const { domain, user, tx } = context.state
|
|
57
|
+
|
|
58
|
+
let results = []
|
|
59
|
+
const _createRecords = patches.filter((patch: any) => patch.cuFlag.toUpperCase() === '+')
|
|
60
|
+
const _updateRecords = patches.filter((patch: any) => patch.cuFlag.toUpperCase() === 'M')
|
|
61
|
+
const pdfReleaseRepo = tx.getRepository(PDFRelease)
|
|
62
|
+
|
|
63
|
+
if (_createRecords.length > 0) {
|
|
64
|
+
for (let i = 0; i < _createRecords.length; i++) {
|
|
65
|
+
const newRecord = _createRecords[i]
|
|
66
|
+
|
|
67
|
+
const result = await pdfReleaseRepo.save({
|
|
68
|
+
...newRecord,
|
|
69
|
+
domain,
|
|
70
|
+
creator: user,
|
|
71
|
+
updater: user
|
|
72
|
+
})
|
|
73
|
+
|
|
74
|
+
results.push({ ...result, cuFlag: '+' })
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
if (_updateRecords.length > 0) {
|
|
79
|
+
for (let i = 0; i < _updateRecords.length; i++) {
|
|
80
|
+
const updateRecord = _updateRecords[i]
|
|
81
|
+
const pdfRelease = await pdfReleaseRepo.findOneBy({ id: updateRecord.id })
|
|
82
|
+
|
|
83
|
+
const result = await pdfReleaseRepo.save({
|
|
84
|
+
...pdfRelease,
|
|
85
|
+
...updateRecord,
|
|
86
|
+
updater: user
|
|
87
|
+
})
|
|
88
|
+
|
|
89
|
+
results.push({ ...result, cuFlag: 'M' })
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
return results
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
@Directive('@transaction')
|
|
97
|
+
@Mutation(returns => Boolean, { description: 'To delete PDFRelease' })
|
|
98
|
+
async deletePDFRelease(@Arg('id') id: string, @Ctx() context: ResolverContext): Promise<boolean> {
|
|
99
|
+
const { domain, tx } = context.state
|
|
100
|
+
|
|
101
|
+
await tx.getRepository(PDFRelease).delete({ domain: { id: domain.id }, id })
|
|
102
|
+
|
|
103
|
+
return true
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
@Directive('@transaction')
|
|
107
|
+
@Mutation(returns => Boolean, { description: 'To delete multiple PDFReleases' })
|
|
108
|
+
async deletePDFReleases(
|
|
109
|
+
@Arg('ids', type => [String]) ids: string[],
|
|
110
|
+
@Ctx() context: ResolverContext
|
|
111
|
+
): Promise<boolean> {
|
|
112
|
+
const { domain, tx } = context.state
|
|
113
|
+
|
|
114
|
+
await tx.getRepository(PDFRelease).delete({
|
|
115
|
+
domain: { id: domain.id },
|
|
116
|
+
id: In(ids)
|
|
117
|
+
})
|
|
118
|
+
|
|
119
|
+
return true
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
@Directive('@transaction')
|
|
123
|
+
@Mutation(returns => Boolean, { description: 'To import multiple PDFReleases' })
|
|
124
|
+
async importPDFReleases(
|
|
125
|
+
@Arg('pdfReleases', type => [PDFReleasePatch]) pdfReleases: PDFReleasePatch[],
|
|
126
|
+
@Ctx() context: ResolverContext
|
|
127
|
+
): Promise<boolean> {
|
|
128
|
+
const { domain, tx } = context.state
|
|
129
|
+
|
|
130
|
+
await Promise.all(
|
|
131
|
+
pdfReleases.map(async (pdfRelease: PDFReleasePatch) => {
|
|
132
|
+
const createdPDFRelease: PDFRelease = await tx.getRepository(PDFRelease).save({ domain, ...pdfRelease })
|
|
133
|
+
})
|
|
134
|
+
)
|
|
135
|
+
|
|
136
|
+
return true
|
|
137
|
+
}
|
|
138
|
+
}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Resolver, Query, FieldResolver, Root, Args, Arg, Ctx, Directive } from 'type-graphql'
|
|
2
|
+
import { Domain, getQueryBuilderFromListParams, getRepository, ListParam } from '@things-factory/shell'
|
|
3
|
+
import { User } from '@things-factory/auth-base'
|
|
4
|
+
import { PDFRelease } from './pdf-release'
|
|
5
|
+
import { PDFReleaseList } from './pdf-release-type'
|
|
6
|
+
|
|
7
|
+
@Resolver(PDFRelease)
|
|
8
|
+
export class PDFReleaseQuery {
|
|
9
|
+
@Query(returns => PDFRelease!, { nullable: true, description: 'To fetch a PDFRelease' })
|
|
10
|
+
async pdfRelease(@Arg('id') id: string, @Ctx() context: ResolverContext): Promise<PDFRelease> {
|
|
11
|
+
const { domain } = context.state
|
|
12
|
+
|
|
13
|
+
return await getRepository(PDFRelease).findOne({
|
|
14
|
+
where: { domain: { id: domain.id }, id }
|
|
15
|
+
})
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
@Query(returns => PDFReleaseList, { description: 'To fetch multiple PDFReleases' })
|
|
19
|
+
async pdfReleases(
|
|
20
|
+
@Args(type => ListParam) params: ListParam,
|
|
21
|
+
@Ctx() context: ResolverContext
|
|
22
|
+
): Promise<PDFReleaseList> {
|
|
23
|
+
const { domain } = context.state
|
|
24
|
+
|
|
25
|
+
const queryBuilder = getQueryBuilderFromListParams({
|
|
26
|
+
domain,
|
|
27
|
+
params,
|
|
28
|
+
repository: await getRepository(PDFRelease),
|
|
29
|
+
searchables: ['name', 'description']
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
const [items, total] = await queryBuilder.getManyAndCount()
|
|
33
|
+
|
|
34
|
+
return { items, total }
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
@FieldResolver(type => Domain)
|
|
38
|
+
async domain(@Root() pdfRelease: PDFRelease): Promise<Domain> {
|
|
39
|
+
return pdfRelease.domainId && (await getRepository(Domain).findOneBy({ id: pdfRelease.domainId }))
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
@FieldResolver(type => User)
|
|
43
|
+
async updater(@Root() pdfRelease: PDFRelease): Promise<User> {
|
|
44
|
+
return pdfRelease.updaterId && (await getRepository(User).findOneBy({ id: pdfRelease.updaterId }))
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
@FieldResolver(type => User)
|
|
48
|
+
async creator(@Root() pdfRelease: PDFRelease): Promise<User> {
|
|
49
|
+
return pdfRelease.creatorId && (await getRepository(User).findOneBy({ id: pdfRelease.creatorId }))
|
|
50
|
+
}
|
|
51
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import type { FileUpload } from 'graphql-upload/GraphQLUpload.js'
|
|
2
|
+
import GraphQLUpload from 'graphql-upload/GraphQLUpload.js'
|
|
3
|
+
import { ObjectType, Field, InputType, Int, ID, registerEnumType } from 'type-graphql'
|
|
4
|
+
|
|
5
|
+
import { ObjectRef, ScalarObject } from '@things-factory/shell'
|
|
6
|
+
|
|
7
|
+
import { PDFRelease, PDFReleaseStatus } from './pdf-release'
|
|
8
|
+
|
|
9
|
+
@InputType()
|
|
10
|
+
export class NewPDFRelease {
|
|
11
|
+
@Field()
|
|
12
|
+
name: string
|
|
13
|
+
|
|
14
|
+
@Field({ nullable: true })
|
|
15
|
+
description?: string
|
|
16
|
+
|
|
17
|
+
@Field(type => PDFReleaseStatus, { nullable: true })
|
|
18
|
+
state?: PDFReleaseStatus
|
|
19
|
+
|
|
20
|
+
@Field({ nullable: true })
|
|
21
|
+
active?: boolean
|
|
22
|
+
|
|
23
|
+
@Field({ nullable: true })
|
|
24
|
+
params?: string
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
@InputType()
|
|
28
|
+
export class PDFReleasePatch {
|
|
29
|
+
@Field(type => ID, { nullable: true })
|
|
30
|
+
id?: string
|
|
31
|
+
|
|
32
|
+
@Field({ nullable: true })
|
|
33
|
+
name?: string
|
|
34
|
+
|
|
35
|
+
@Field({ nullable: true })
|
|
36
|
+
description?: string
|
|
37
|
+
|
|
38
|
+
@Field(type => PDFReleaseStatus, { nullable: true })
|
|
39
|
+
state?: PDFReleaseStatus
|
|
40
|
+
|
|
41
|
+
@Field({ nullable: true })
|
|
42
|
+
active?: boolean
|
|
43
|
+
|
|
44
|
+
@Field({ nullable: true })
|
|
45
|
+
cuFlag?: string
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
@ObjectType()
|
|
49
|
+
export class PDFReleaseList {
|
|
50
|
+
@Field(type => [PDFRelease])
|
|
51
|
+
items: PDFRelease[]
|
|
52
|
+
|
|
53
|
+
@Field(type => Int)
|
|
54
|
+
total: number
|
|
55
|
+
}
|