@things-factory/dataset 5.0.0-alpha.2 → 5.0.0-alpha.22

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 (96) hide show
  1. package/README.md +13 -0
  2. package/assets/data-samples.jpg +0 -0
  3. package/client/bootstrap.js +16 -1
  4. package/client/pages/data-entry-form.js +84 -0
  5. package/client/pages/data-item-list.js +58 -16
  6. package/client/pages/data-ooc-view.js +182 -0
  7. package/client/pages/data-ooc.js +487 -0
  8. package/client/pages/data-sample-view.js +97 -0
  9. package/client/pages/data-sample.js +149 -56
  10. package/client/pages/data-sensor.js +8 -18
  11. package/client/pages/data-set.js +200 -75
  12. package/client/route.js +4 -0
  13. package/dist-server/controllers/create-data-sample.js +124 -0
  14. package/dist-server/controllers/create-data-sample.js.map +1 -0
  15. package/dist-server/controllers/data-use-case.js +57 -0
  16. package/dist-server/controllers/data-use-case.js.map +1 -0
  17. package/dist-server/controllers/index.js +17 -0
  18. package/dist-server/controllers/index.js.map +1 -1
  19. package/dist-server/index.js +2 -0
  20. package/dist-server/index.js.map +1 -1
  21. package/dist-server/routes.js +9 -24
  22. package/dist-server/routes.js.map +1 -1
  23. package/dist-server/service/data-item/data-item-mutation.js +5 -1
  24. package/dist-server/service/data-item/data-item-mutation.js.map +1 -1
  25. package/dist-server/service/data-item/data-item-type.js +14 -5
  26. package/dist-server/service/data-item/data-item-type.js.map +1 -1
  27. package/dist-server/service/data-item/data-item.js +20 -7
  28. package/dist-server/service/data-item/data-item.js.map +1 -1
  29. package/dist-server/service/data-ooc/data-ooc-mutation.js +92 -0
  30. package/dist-server/service/data-ooc/data-ooc-mutation.js.map +1 -0
  31. package/dist-server/service/data-ooc/data-ooc-query.js +115 -0
  32. package/dist-server/service/data-ooc/data-ooc-query.js.map +1 -0
  33. package/dist-server/service/data-ooc/data-ooc-subscription.js +65 -0
  34. package/dist-server/service/data-ooc/data-ooc-subscription.js.map +1 -0
  35. package/dist-server/service/data-ooc/data-ooc-type.js +107 -0
  36. package/dist-server/service/data-ooc/data-ooc-type.js.map +1 -0
  37. package/dist-server/service/data-ooc/data-ooc.js +237 -0
  38. package/dist-server/service/data-ooc/data-ooc.js.map +1 -0
  39. package/dist-server/service/data-ooc/index.js +10 -0
  40. package/dist-server/service/data-ooc/index.js.map +1 -0
  41. package/dist-server/service/data-sample/data-sample-mutation.js +3 -105
  42. package/dist-server/service/data-sample/data-sample-mutation.js.map +1 -1
  43. package/dist-server/service/data-sample/data-sample-type.js +13 -42
  44. package/dist-server/service/data-sample/data-sample-type.js.map +1 -1
  45. package/dist-server/service/data-sample/data-sample.js +40 -9
  46. package/dist-server/service/data-sample/data-sample.js.map +1 -1
  47. package/dist-server/service/data-set/data-set-mutation.js +1 -2
  48. package/dist-server/service/data-set/data-set-mutation.js.map +1 -1
  49. package/dist-server/service/data-set/data-set-query.js +12 -0
  50. package/dist-server/service/data-set/data-set-query.js.map +1 -1
  51. package/dist-server/service/data-set/data-set-type.js +19 -2
  52. package/dist-server/service/data-set/data-set-type.js.map +1 -1
  53. package/dist-server/service/data-set/data-set.js +22 -10
  54. package/dist-server/service/data-set/data-set.js.map +1 -1
  55. package/dist-server/service/data-spec/data-spec-manager.js +20 -0
  56. package/dist-server/service/data-spec/data-spec-manager.js.map +1 -0
  57. package/dist-server/service/data-spec/data-spec-query.js +48 -0
  58. package/dist-server/service/data-spec/data-spec-query.js.map +1 -0
  59. package/dist-server/service/data-spec/data-spec.js +78 -0
  60. package/dist-server/service/data-spec/data-spec.js.map +1 -0
  61. package/dist-server/service/data-spec/index.js +8 -0
  62. package/dist-server/service/data-spec/index.js.map +1 -0
  63. package/dist-server/service/index.js +12 -4
  64. package/dist-server/service/index.js.map +1 -1
  65. package/package.json +15 -12
  66. package/server/controllers/create-data-sample.ts +166 -0
  67. package/server/controllers/data-use-case.ts +85 -0
  68. package/server/controllers/index.ts +1 -0
  69. package/server/index.ts +3 -0
  70. package/server/routes.ts +17 -31
  71. package/server/service/data-item/data-item-mutation.ts +6 -1
  72. package/server/service/data-item/data-item-type.ts +11 -7
  73. package/server/service/data-item/data-item.ts +16 -6
  74. package/server/service/data-ooc/data-ooc-mutation.ts +150 -0
  75. package/server/service/data-ooc/data-ooc-query.ts +63 -0
  76. package/server/service/data-ooc/data-ooc-subscription.ts +51 -0
  77. package/server/service/data-ooc/data-ooc-type.ts +68 -0
  78. package/server/service/data-ooc/data-ooc.ts +204 -0
  79. package/server/service/data-ooc/index.ts +7 -0
  80. package/server/service/data-sample/data-sample-mutation.ts +6 -128
  81. package/server/service/data-sample/data-sample-type.ts +8 -29
  82. package/server/service/data-sample/data-sample.ts +36 -6
  83. package/server/service/data-set/data-set-mutation.ts +1 -4
  84. package/server/service/data-set/data-set-query.ts +8 -1
  85. package/server/service/data-set/data-set-type.ts +17 -6
  86. package/server/service/data-set/data-set.ts +18 -8
  87. package/server/service/data-spec/data-spec-manager.ts +21 -0
  88. package/server/service/data-spec/data-spec-query.ts +21 -0
  89. package/server/service/data-spec/data-spec.ts +44 -0
  90. package/server/service/data-spec/index.ts +5 -0
  91. package/server/service/index.ts +16 -8
  92. package/things-factory.config.js +4 -0
  93. package/translations/en.json +21 -0
  94. package/translations/ko.json +22 -1
  95. package/translations/ms.json +21 -0
  96. package/translations/zh.json +21 -0
@@ -1,7 +1,7 @@
1
1
  import { Arg, Args, Ctx, Directive, FieldResolver, Query, Resolver, Root } from 'type-graphql'
2
2
  import { getRepository } from 'typeorm'
3
3
 
4
- import { User } from '@things-factory/auth-base'
4
+ import { Role, User } from '@things-factory/auth-base'
5
5
  import { convertListParams, Domain, ListParam } from '@things-factory/shell'
6
6
 
7
7
  import { DataItem } from '../data-item/data-item'
@@ -38,6 +38,13 @@ export class DataSetQuery {
38
38
  })
39
39
  }
40
40
 
41
+ @FieldResolver(type => Role)
42
+ async supervisoryRole(@Root() dataSet: DataSet): Promise<Role> {
43
+ return await getRepository(Role).findOne({
44
+ id: dataSet.supervisoryRoleId
45
+ })
46
+ }
47
+
41
48
  @FieldResolver(type => Domain)
42
49
  async domain(@Root() dataSet: DataSet): Promise<Domain> {
43
50
  return await getRepository(Domain).findOne(dataSet.domainId)
@@ -1,6 +1,5 @@
1
1
  import { Field, ID, InputType, Int, ObjectType } from 'type-graphql'
2
-
3
- import { ScalarObject } from '@things-factory/shell'
2
+ import { ObjectRef, ScalarObject } from '@things-factory/shell'
4
3
 
5
4
  import { DataSet } from './data-set'
6
5
 
@@ -12,14 +11,20 @@ export class NewDataSet {
12
11
  @Field({ nullable: true })
13
12
  description?: string
14
13
 
14
+ @Field(type => ObjectRef, { nullable: true })
15
+ supervisoryRole?: ObjectRef
16
+
15
17
  @Field({ nullable: true })
16
18
  active?: boolean
17
19
 
18
20
  @Field({ nullable: true })
19
- type?: string
21
+ type?: 'manual' | 'automatic' | string
22
+
23
+ @Field({ nullable: true })
24
+ useCase?: string
20
25
 
21
26
  @Field(type => ScalarObject, { nullable: true })
22
- partitionKeys?: object
27
+ partitionKeys?: ScalarObject
23
28
 
24
29
  @Field({ nullable: true })
25
30
  schedule?: string
@@ -39,14 +44,20 @@ export class DataSetPatch {
39
44
  @Field({ nullable: true })
40
45
  description?: string
41
46
 
47
+ @Field(type => ObjectRef, { nullable: true })
48
+ supervisoryRole?: ObjectRef
49
+
42
50
  @Field({ nullable: true })
43
51
  active?: boolean
44
52
 
45
53
  @Field({ nullable: true })
46
- type?: string
54
+ type?: 'manual' | 'automatic' | string
55
+
56
+ @Field({ nullable: true })
57
+ useCase?: string
47
58
 
48
59
  @Field(type => ScalarObject, { nullable: true })
49
- partitionKeys?: object
60
+ partitionKeys?: ScalarObject
50
61
 
51
62
  @Field({ nullable: true })
52
63
  schedule?: string
@@ -11,11 +11,10 @@ import {
11
11
  UpdateDateColumn
12
12
  } from 'typeorm'
13
13
 
14
- import { User } from '@things-factory/auth-base'
14
+ import { Role, User } from '@things-factory/auth-base'
15
15
  import { Domain, ScalarObject } from '@things-factory/shell'
16
16
 
17
17
  import { DataItem } from '../data-item/data-item'
18
- import { DataSample } from '../data-sample/data-sample'
19
18
 
20
19
  @Entity()
21
20
  @Index('ix_data_set_0', (dataSet: DataSet) => [dataSet.domain, dataSet.name], { unique: true })
@@ -48,15 +47,30 @@ export class DataSet {
48
47
  @Field({ nullable: true })
49
48
  active?: boolean
50
49
 
50
+ @ManyToOne(type => Role, {
51
+ nullable: true
52
+ })
53
+ @Field(type => Role, { nullable: true })
54
+ supervisoryRole?: Role
55
+
56
+ @RelationId((dataSet: DataSet) => dataSet.supervisoryRole)
57
+ supervisoryRoleId?: string
58
+
59
+ @Column({
60
+ nullable: true
61
+ })
62
+ @Field({ nullable: true })
63
+ type?: 'manual' | 'automatic' | string
64
+
51
65
  @Column({
52
66
  nullable: true
53
67
  })
54
68
  @Field({ nullable: true })
55
- type?: string
69
+ useCase?: string
56
70
 
57
71
  @Column('simple-json', { nullable: true })
58
72
  @Field(type => ScalarObject, { nullable: true })
59
- partitionKeys?: object
73
+ partitionKeys?: ScalarObject
60
74
 
61
75
  @Column({
62
76
  nullable: true
@@ -68,10 +82,6 @@ export class DataSet {
68
82
  @Field(type => [DataItem], { nullable: true })
69
83
  dataItems: DataItem[]
70
84
 
71
- @OneToMany(type => DataSample, dataSample => dataSample.dataSet)
72
- @Field(type => [DataSample], { nullable: true })
73
- dataSamples: DataSample[]
74
-
75
85
  @Column({
76
86
  nullable: true
77
87
  })
@@ -0,0 +1,21 @@
1
+ import { DataSpec } from './data-spec'
2
+
3
+ export class DataSpecManager {
4
+ static specRegistry: { [type: string]: DataSpec } = {}
5
+
6
+ static getDataSpec(type: string): DataSpec {
7
+ return DataSpecManager.specRegistry[type]
8
+ }
9
+
10
+ static registerDataSpec(type: string, dataSpec: DataSpec) {
11
+ DataSpecManager.specRegistry[type] = dataSpec
12
+ }
13
+
14
+ static unregisterDataSpec(type: string) {
15
+ delete DataSpecManager.specRegistry[type]
16
+ }
17
+
18
+ static getDataSpecs(): DataSpec[] {
19
+ return Object.values(DataSpecManager.specRegistry)
20
+ }
21
+ }
@@ -0,0 +1,21 @@
1
+ import { Resolver, Query, Arg } from 'type-graphql'
2
+ import { DataSpec, DataSpecList } from './data-spec'
3
+ import { DataSpecManager } from './data-spec-manager'
4
+
5
+ @Resolver(DataSpec)
6
+ export class DataSpecQuery {
7
+ @Query(returns => DataSpec, { description: 'To fetch a data-spec' })
8
+ dataSpec(@Arg('name') name: string): DataSpec {
9
+ return DataSpecManager.getDataSpec(name)
10
+ }
11
+
12
+ @Query(returns => DataSpecList, { description: 'To fetch all data-specs' })
13
+ dataSpecs(): DataSpecList {
14
+ const items = DataSpecManager.getDataSpecs()
15
+
16
+ return {
17
+ items,
18
+ total: items.length
19
+ }
20
+ }
21
+ }
@@ -0,0 +1,44 @@
1
+ import { ObjectType, Field, Int } from 'type-graphql'
2
+ import { ScalarObject } from '@things-factory/shell'
3
+
4
+ @ObjectType()
5
+ export class DataSpecItem {
6
+ @Field()
7
+ type: string
8
+
9
+ @Field()
10
+ label: string
11
+
12
+ @Field()
13
+ name: string
14
+
15
+ @Field({ nullable: true })
16
+ placeholder?: string
17
+
18
+ @Field(type => ScalarObject, { nullable: true })
19
+ property?: ScalarObject
20
+ }
21
+
22
+ @ObjectType()
23
+ export class DataSpec {
24
+ @Field()
25
+ name: string
26
+
27
+ @Field({ nullable: true })
28
+ description: string
29
+
30
+ @Field({ nullable: true })
31
+ help: string
32
+
33
+ @Field(type => [DataSpecItem], { nullable: true })
34
+ specItems: DataSpecItem[]
35
+ }
36
+
37
+ @ObjectType()
38
+ export class DataSpecList {
39
+ @Field(type => [DataSpec])
40
+ items: DataSpec[]
41
+
42
+ @Field(type => Int)
43
+ total: number
44
+ }
@@ -0,0 +1,5 @@
1
+ import { DataSpec } from './data-spec'
2
+ import { DataSpecQuery } from './data-spec-query'
3
+
4
+ export const entities = [DataSpec]
5
+ export const resolvers = [DataSpecQuery]
@@ -1,29 +1,37 @@
1
+ import { entities as DataItemEntities, resolvers as DataItemResolvers } from './data-item'
1
2
  /* IMPORT ENTITIES AND RESOLVERS */
2
- import { entities as DataSensorEntities, resolvers as DataSensorResolvers } from './data-sensor'
3
+ import { entities as DataOocEntities, resolvers as DataOocResolvers } from './data-ooc'
3
4
  import { entities as DataSampleEntities, resolvers as DataSampleResolvers } from './data-sample'
4
- import { entities as DataItemEntities, resolvers as DataItemResolvers } from './data-item'
5
+ import { entities as DataSensorEntities, resolvers as DataSensorResolvers } from './data-sensor'
5
6
  import { entities as DataSetEntities, resolvers as DataSetResolvers } from './data-set'
7
+ import { entities as DataSpecEntities, resolvers as DataSpecResolvers } from './data-spec'
6
8
 
7
9
  /* EXPORT ENTITY TYPES */
10
+ export * from './data-ooc/data-ooc'
8
11
  export * from './data-sensor/data-sensor'
9
12
  export * from './data-sample/data-sample'
10
13
  export * from './data-item/data-item'
11
14
  export * from './data-set/data-set'
15
+ export * from './data-spec/data-spec'
12
16
 
13
17
  export const entities = [
14
18
  /* ENTITIES */
15
- ...DataSensorEntities,
16
- ...DataSampleEntities,
19
+ ...DataOocEntities,
20
+ ...DataSensorEntities,
21
+ ...DataSampleEntities,
17
22
  ...DataItemEntities,
18
- ...DataSetEntities
23
+ ...DataSetEntities,
24
+ ...DataSpecEntities
19
25
  ]
20
26
 
21
27
  export const schema = {
22
28
  resolverClasses: [
23
29
  /* RESOLVER CLASSES */
24
- ...DataSensorResolvers,
25
- ...DataSampleResolvers,
30
+ ...DataOocResolvers,
31
+ ...DataSensorResolvers,
32
+ ...DataSampleResolvers,
26
33
  ...DataItemResolvers,
27
- ...DataSetResolvers
34
+ ...DataSetResolvers,
35
+ ...DataSpecResolvers
28
36
  ]
29
37
  }
@@ -15,6 +15,10 @@ export default {
15
15
  {
16
16
  tagname: 'data-sample-page',
17
17
  page: 'data-sample'
18
+ },
19
+ {
20
+ tagname: 'data-ooc-page',
21
+ page: 'data-ooc'
18
22
  }
19
23
  ],
20
24
  bootstrap
@@ -1,19 +1,40 @@
1
1
  {
2
+ "button.corrected": "corrected",
3
+ "button.reviewed": "reviewed",
2
4
  "field.appliance": "appliance",
3
5
  "field.collected_at": "collected at",
6
+ "field.corrective-action": "corrective action",
7
+ "field.corrector": "corrector",
8
+ "field.corrected-at": "corrected at",
4
9
  "field.data": "data",
10
+ "field.data-sample": "data sample",
5
11
  "field.data-set": "data set",
6
12
  "field.device-id": "device id",
7
13
  "field.netmask": "network mask",
14
+ "field.oos": "out of critical limit",
15
+ "field.ooc": "out of control limit",
16
+ "field.options": "options",
8
17
  "field.partition-keys": "partition keys",
9
18
  "field.quota": "sampling #",
10
19
  "field.raw-data": "raw data",
11
20
  "field.ref-by": "ref. by",
12
21
  "field.serial-no": "serial #",
13
22
  "field.spec": "spec",
23
+ "field.state": "state",
24
+ "field.supervisory-role": "supervisory role",
14
25
  "field.tag": "tag name",
26
+ "field.unit": "unit",
27
+ "field.use-case": "use case",
28
+ "field.work-date": "work date",
29
+ "field.work-shift": "work shift",
30
+ "text.data sample created successfully": "a data sample created successfully",
31
+ "text.data ooc updated successfully": "a data ooc updated successfully",
32
+ "title.data-entry-form": "data entry form",
15
33
  "title.data-item list": "data item list",
34
+ "title.data-ooc list": "data OOC list",
35
+ "title.data-ooc view": "data OOC view",
16
36
  "title.data-sample list": "data sample list",
37
+ "title.data-sample view": "data sample view",
17
38
  "title.data-sensor list": "data sensor list",
18
39
  "title.data-set list": "data set list"
19
40
  }
@@ -1,19 +1,40 @@
1
1
  {
2
+ "button.corrected": "조치완료",
3
+ "button.reviewed": "검토완료",
2
4
  "field.appliance": "어플라이언스",
3
5
  "field.collected_at": "수집일시",
6
+ "field.corrective-action": "조치 활동",
7
+ "field.corrector": "조치자",
8
+ "field.corrected-at": "조치 시간",
4
9
  "field.data": "데이타",
10
+ "field.data-sample": "데이타 샘플",
5
11
  "field.data-set": "데이타셋",
6
12
  "field.device-id": "디바이스 아이디",
7
13
  "field.netmask": "네트워크마스크",
14
+ "field.oos": "허용한계 이탈여부",
15
+ "field.ooc": "관리한계 이탈여부",
16
+ "field.options": "선택옵션",
8
17
  "field.partition-keys": "파티션 키",
9
18
  "field.quota": "샘플수",
10
19
  "field.raw-data": "데이타 원본",
11
20
  "field.ref-by": "참조아이템",
12
21
  "field.serial-no": "시리얼번호",
13
22
  "field.spec": "명세",
14
- "field.tag": "태크이름",
23
+ "field.state": "상태",
24
+ "field.supervisory-role": "관리자 역할",
25
+ "field.tag": "태그이름",
26
+ "field.unit": "단위",
27
+ "field.use-case": "사용 사례",
28
+ "field.work-date": "작업기준일",
29
+ "field.work-shift": "교대근무조",
30
+ "text.data sample created successfully": "데이타 샘플이 성공적으로 생성되었습니다",
31
+ "text.data ooc updated successfully": "이탈데이타 내용이 수정되었습니다",
32
+ "title.data-entry-form": "데이타 입력",
15
33
  "title.data-item list": "데이타 아이템 조회",
34
+ "title.data-ooc list": "데이타 이탈점 조회",
35
+ "title.data-ooc view": "데이타 이탈점 상세",
16
36
  "title.data-sample list": "데이타 샘플 조회",
37
+ "title.data-sample view": "데이타 샘플 상세",
17
38
  "title.data-sensor list": "데이타 센서 조회",
18
39
  "title.data-set list": "데이타 셋 조회"
19
40
  }
@@ -1,19 +1,40 @@
1
1
  {
2
+ "button.corrected": "corrected",
3
+ "button.reviewed": "reviewed",
2
4
  "field.appliance": "appliance",
3
5
  "field.collected_at": "collected at",
6
+ "field.corrective-action": "corrective action",
7
+ "field.corrector": "corrector",
8
+ "field.corrected-at": "corrected at",
4
9
  "field.data": "data",
10
+ "field.data-sample": "data sample",
5
11
  "field.data-set": "data set",
6
12
  "field.device-id": "device id",
7
13
  "field.netmask": "network mask",
14
+ "field.oos": "out of critical limit",
15
+ "field.ooc": "out of control limit",
16
+ "field.options": "options",
8
17
  "field.partition-keys": "partition keys",
9
18
  "field.quota": "sampling #",
10
19
  "field.raw-data": "raw data",
11
20
  "field.ref-by": "ref. by",
12
21
  "field.serial-no": "serial #",
13
22
  "field.spec": "spec",
23
+ "field.state": "keadaan",
24
+ "field.supervisory-role": "supervisory role",
14
25
  "field.tag": "tag name",
26
+ "field.unit": "unit",
27
+ "field.use-case": "use case",
28
+ "field.work-date": "work date",
29
+ "field.work-shift": "work shift",
30
+ "text.data sample created successfully": "a data sample created successfully",
31
+ "text.data ooc updated successfully": "a data ooc updated successfully",
32
+ "title.data-entry-form": "data entry form",
15
33
  "title.data-item list": "data item list",
34
+ "title.data-ooc list": "data OOC list",
35
+ "title.data-ooc view": "data OOC view",
16
36
  "title.data-sample list": "data sample list",
37
+ "title.data-sample view": "data sample view",
17
38
  "title.data-sensor list": "data sensor list",
18
39
  "title.data-set list": "data set list"
19
40
  }
@@ -1,19 +1,40 @@
1
1
  {
2
+ "button.corrected": "corrected",
3
+ "button.reviewed": "reviewed",
2
4
  "field.appliance": "appliance",
3
5
  "field.collected_at": "collected at",
6
+ "field.corrective-action": "corrective action",
7
+ "field.corrector": "corrector",
8
+ "field.corrected-at": "corrected at",
4
9
  "field.data": "data",
10
+ "field.data-sample": "data sample",
5
11
  "field.data-set": "data set",
6
12
  "field.device-id": "device id",
7
13
  "field.netmask": "network mask",
14
+ "field.oos": "out of critical limit",
15
+ "field.ooc": "out of control limit",
16
+ "field.options": "options",
8
17
  "field.partition-keys": "partition keys",
9
18
  "field.quota": "sampling #",
10
19
  "field.raw-data": "raw data",
11
20
  "field.ref-by": "ref. by",
12
21
  "field.serial-no": "serial #",
13
22
  "field.spec": "spec",
23
+ "field.state": "状态",
24
+ "field.supervisory-role": "supervisory role",
14
25
  "field.tag": "tag name",
26
+ "field.unit": "unit",
27
+ "field.use-case": "use case",
28
+ "field.work-date": "work date",
29
+ "field.work-shift": "work shift",
30
+ "text.data sample created successfully": "a data sample created successfully",
31
+ "text.data ooc updated successfully": "a data ooc updated successfully",
32
+ "title.data-entry-form": "data entry form",
15
33
  "title.data-item list": "data item list",
34
+ "title.data-ooc list": "data OOC list",
35
+ "title.data-ooc view": "data OOC view",
16
36
  "title.data-sample list": "data sample list",
37
+ "title.data-sample view": "data sample view",
17
38
  "title.data-sensor list": "data sensor list",
18
39
  "title.data-set list": "data set list"
19
40
  }