diff --git a/packages/threat-composer-app/src/utils/convertToDocx/components/Table.ts b/packages/threat-composer-app/src/utils/convertToDocx/components/Table.ts new file mode 100644 index 00000000..b4b61e8e --- /dev/null +++ b/packages/threat-composer-app/src/utils/convertToDocx/components/Table.ts @@ -0,0 +1,32 @@ +/** ******************************************************************************************************************* + Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + + Licensed under the Apache License, Version 2.0 (the "License"). + You may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + ******************************************************************************************************************** */ +import { Table as DocxTable, ITableOptions } from 'docx'; + +class Table extends DocxTable { + constructor(opts: ITableOptions) { + super({ + ...opts, + margins: { + top: 16, + bottom: 16, + left: 16, + right: 16, + }, + }); + } +} + +export default Table; \ No newline at end of file diff --git a/packages/threat-composer-app/src/utils/convertToDocx/components/TableHeaderCell.ts b/packages/threat-composer-app/src/utils/convertToDocx/components/TableHeaderCell.ts new file mode 100644 index 00000000..7d133c69 --- /dev/null +++ b/packages/threat-composer-app/src/utils/convertToDocx/components/TableHeaderCell.ts @@ -0,0 +1,31 @@ +/** ******************************************************************************************************************* + Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + + Licensed under the Apache License, Version 2.0 (the "License"). + You may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + ******************************************************************************************************************** */ +import { TableCell, Paragraph, IParagraphOptions } from 'docx'; + +class TableHeaderCell extends TableCell { + constructor(paragraph: string | IParagraphOptions) { + super({ + children: [new Paragraph(paragraph)], + verticalAlign: 'center', + shading: { + fill: '000000', + color: 'FFFFFF', + }, + }); + } +} + +export default TableHeaderCell; \ No newline at end of file diff --git a/packages/threat-composer-app/src/utils/convertToDocx/config.ts b/packages/threat-composer-app/src/utils/convertToDocx/config.ts index 5de7f5aa..15e99c91 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/config.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/config.ts @@ -85,4 +85,15 @@ export const DEFAULT_NUMBERINGS: ILevelsOptions[] = [ }, }, }, -]; \ No newline at end of file +]; + +export const PT_BASE = 20; +export const LINE_BASE = 276; +export const SPACING = { + line: LINE_BASE, + after: PT_BASE * 6, +}; +export const LIST_PARA_SPACING = { + line: Math.floor(LINE_BASE * 0.9), + after: PT_BASE * 3, +}; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/convertMarkdown/transformer.ts b/packages/threat-composer-app/src/utils/convertToDocx/convertMarkdown/transformer.ts index ec4776d3..f580e8e4 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/convertMarkdown/transformer.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/convertMarkdown/transformer.ts @@ -19,7 +19,6 @@ import { Document, Paragraph, ParagraphChild, - Table, TableRow, TableCell, TableOfContents, @@ -37,6 +36,8 @@ import { import type { IPropertiesOptions } from 'docx/build/file/core-properties'; import type * as mdast from './mdast'; import { invariant } from './utils'; +import Table from '../components/Table'; +import TableHeaderCell from '../components/TableHeaderCell'; const ORDERED_LIST_REF = 'ordered'; const INDENT = 0.5; @@ -339,6 +340,7 @@ const buildParagraph = ({ children }: mdast.Paragraph, ctx: Context) => { } return new Paragraph({ children: nodes, + style: 'normalPara', indent: ctx.indent > 0 ? { @@ -352,11 +354,13 @@ const buildParagraph = ({ children }: mdast.Paragraph, ctx: Context) => { reference: ORDERED_LIST_REF, level: list.level, }, + style: 'listPara', } : { bullet: { level: list.level, }, + style: 'listPara', })), }); }; @@ -443,12 +447,36 @@ const buildTable = ({ children, align }: mdast.Table, ctx: Context) => { }); return new Table({ - rows: children.map((r) => { - return buildTableRow(r, ctx, cellAligns); + rows: children.map((r, index) => { + return index === 0 ? buildTableHeaderRow(r, ctx, cellAligns) : buildTableRow(r, ctx, cellAligns); }), }); }; +const buildTableHeaderRow = ( + { children }: mdast.TableRow, + ctx: Context, + cellAligns: any[] | undefined, +) => { + return new TableRow({ + children: children.map((c, i) => { + return buildTableHeaderCell(c, ctx, cellAligns?.[i]); + }), + }); +}; + +const buildTableHeaderCell = ( + { children }: mdast.TableCell, + ctx: Context, + align: any | undefined, +) => { + const { nodes } = convertNodes(children, ctx); + return new TableHeaderCell({ + alignment: align, + children: nodes, + }); +}; + const buildTableRow = ( { children }: mdast.TableRow, ctx: Context, diff --git a/packages/threat-composer-app/src/utils/convertToDocx/fetchImage.ts b/packages/threat-composer-app/src/utils/convertToDocx/fetchImage.ts index a46f653d..5ef1f236 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/fetchImage.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/fetchImage.ts @@ -17,7 +17,8 @@ const WORD_DOCX_WIDTH = 600; const fetchImage = async ( url: string, -): Promise<{ image: ArrayBuffer; width: number; height: number }> => { + fetchOriginalFailed?: boolean, +): Promise<{ image: ArrayBuffer; width: number; height: number; fetchOriginalFailed: boolean }> => { const image = new Image(); try { const res = await fetch(url); @@ -32,6 +33,7 @@ const fetchImage = async ( image: buf, width, height, + fetchOriginalFailed: fetchOriginalFailed || false, }); }; image.onerror = reject; @@ -39,7 +41,7 @@ const fetchImage = async ( }); } catch (e) { console.log('Failed to fetch image and returns placeholder image', e); - return fetchImage(''); + return fetchImage('', true); } }; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getArchitecture.ts b/packages/threat-composer-app/src/utils/convertToDocx/getArchitecture.ts index c7fa2825..a2bdb547 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getArchitecture.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getArchitecture.ts @@ -14,9 +14,9 @@ limitations under the License. ******************************************************************************************************************** */ import { DataExchangeFormat } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, ImageRun } from 'docx'; +import { Paragraph, HeadingLevel, TextRun } from 'docx'; import convertMarkdown from './convertMarkdown'; -import fetchImage from './fetchImage'; +import getImage from './getImage'; const getArchitecture = async ( data: DataExchangeFormat, @@ -51,19 +51,8 @@ const getArchitecture = async ( ], })); - const image = await fetchImage(data.architecture.image); - - children.push(new Paragraph({ - children: [ - new ImageRun({ - data: image.image, - transformation: { - width: image.width, - height: image.height, - }, - }), - ], - })); + const image = await getImage(data.architecture.image); + children.push(image); } } diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getAssets.ts b/packages/threat-composer-app/src/utils/convertToDocx/getAssets.ts index 6aed8eb9..b18e63be 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getAssets.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getAssets.ts @@ -14,7 +14,8 @@ limitations under the License. ******************************************************************************************************************** */ import { TemplateThreatStatement, DataExchangeFormat, standardizeNumericId } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, Table, TableCell, TableRow } from 'docx'; +import { Paragraph, HeadingLevel, TextRun, TableCell, TableRow } from 'docx'; +import Table from './components/Table'; import getAnchorLink from './getAnchorLink'; import getBookmark from './getBookmark'; import getHeaderRow from './getHeaderRow'; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getAssumptions.ts b/packages/threat-composer-app/src/utils/convertToDocx/getAssumptions.ts index 957f2c84..0490ddca 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getAssumptions.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getAssumptions.ts @@ -14,7 +14,8 @@ limitations under the License. ******************************************************************************************************************** */ import { Assumption, AssumptionLink, DataExchangeFormat, standardizeNumericId } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, Table, TableCell, TableRow } from 'docx'; +import { Paragraph, HeadingLevel, TextRun, TableCell, TableRow } from 'docx'; +import Table from './components/Table'; import getAnchorLink from './getAnchorLink'; import getBookmark from './getBookmark'; import getHeaderRow from './getHeaderRow'; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getDataflow.ts b/packages/threat-composer-app/src/utils/convertToDocx/getDataflow.ts index 38d3c876..f238b155 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getDataflow.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getDataflow.ts @@ -14,9 +14,9 @@ limitations under the License. ******************************************************************************************************************** */ import { DataExchangeFormat } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, ImageRun } from 'docx'; +import { Paragraph, HeadingLevel, TextRun } from 'docx'; import convertMarkdown from './convertMarkdown'; -import fetchImage from './fetchImage'; +import getImage from './getImage'; const getDataflow = async ( data: DataExchangeFormat, @@ -51,19 +51,8 @@ const getDataflow = async ( ], })); - const image = await fetchImage(data.dataflow.image); - - children.push(new Paragraph({ - children: [ - new ImageRun({ - data: image.image, - transformation: { - width: image.width, - height: image.height, - }, - }), - ], - })); + const image = await getImage(data.dataflow.image); + children.push(image); } } diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getHeaderRow.ts b/packages/threat-composer-app/src/utils/convertToDocx/getHeaderRow.ts index e12c54db..e41f8262 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getHeaderRow.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getHeaderRow.ts @@ -13,13 +13,12 @@ See the License for the specific language governing permissions and limitations under the License. ******************************************************************************************************************** */ -import { Paragraph, TableCell, TableRow } from 'docx'; +import { TableRow } from 'docx'; +import TableHeaderCell from './components/TableHeaderCell'; const getHeaderRow = (headers: string[]) => { return new TableRow({ - children: headers.map(h => new TableCell({ - children: [new Paragraph(h)], - })), + children: headers.map(h => new TableHeaderCell(h)), }); }; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getImage.ts b/packages/threat-composer-app/src/utils/convertToDocx/getImage.ts new file mode 100644 index 00000000..c8ac2afd --- /dev/null +++ b/packages/threat-composer-app/src/utils/convertToDocx/getImage.ts @@ -0,0 +1,54 @@ +/** ******************************************************************************************************************* + Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. + + Licensed under the Apache License, Version 2.0 (the "License"). + You may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + ******************************************************************************************************************** */ +import { ExternalHyperlink, ImageRun, Paragraph } from 'docx'; +import fetchImage from './fetchImage'; + +const getImage = async (imageUrl: string) => { + const image = await fetchImage(imageUrl); + + if (imageUrl.startsWith('https://') || imageUrl.startsWith('http://')) { + return new Paragraph({ + children: [ + new ExternalHyperlink({ + link: imageUrl, + children: [ + new ImageRun({ + data: image.image, + transformation: { + width: image.width, + height: image.height, + }, + }), + ], + }), + ], + }); + } + + return new Paragraph({ + children: [ + new ImageRun({ + data: image.image, + transformation: { + width: image.width, + height: image.height, + }, + }), + ], + }); +}; + +export default getImage; \ No newline at end of file diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getMitigations.ts b/packages/threat-composer-app/src/utils/convertToDocx/getMitigations.ts index 2c6bf72d..2b7fa535 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getMitigations.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getMitigations.ts @@ -14,7 +14,8 @@ limitations under the License. ******************************************************************************************************************** */ import { Mitigation, MitigationLink, AssumptionLink, DataExchangeFormat, standardizeNumericId } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, Table, TableCell, TableRow } from 'docx'; +import { Paragraph, HeadingLevel, TextRun, TableCell, TableRow } from 'docx'; +import Table from './components/Table'; import getAnchorLink from './getAnchorLink'; import getBookmark from './getBookmark'; import getHeaderRow from './getHeaderRow'; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/getThreats.ts b/packages/threat-composer-app/src/utils/convertToDocx/getThreats.ts index c7887dc9..c9149f86 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/getThreats.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/getThreats.ts @@ -14,7 +14,8 @@ limitations under the License. ******************************************************************************************************************** */ import { AssumptionLink, DataExchangeFormat, MitigationLink, TemplateThreatStatement, standardizeNumericId } from '@aws/threat-composer'; -import { Paragraph, HeadingLevel, TextRun, Table, TableRow, TableCell } from 'docx'; +import { Paragraph, HeadingLevel, TextRun, TableRow, TableCell } from 'docx'; +import Table from './components/Table'; import getAnchorLink from './getAnchorLink'; import getBookmark from './getBookmark'; import getHeaderRow from './getHeaderRow'; diff --git a/packages/threat-composer-app/src/utils/convertToDocx/index.ts b/packages/threat-composer-app/src/utils/convertToDocx/index.ts index 5a792847..66b9b659 100644 --- a/packages/threat-composer-app/src/utils/convertToDocx/index.ts +++ b/packages/threat-composer-app/src/utils/convertToDocx/index.ts @@ -15,7 +15,7 @@ ******************************************************************************************************************** */ import { DataExchangeFormat } from '@aws/threat-composer'; import { Document, Packer } from 'docx'; -import { ORDERED_LIST_REF, DEFAULT_NUMBERINGS } from './config'; +import { ORDERED_LIST_REF, DEFAULT_NUMBERINGS, SPACING, LIST_PARA_SPACING } from './config'; import getApplicationInfo from './getApplicationInfo'; import { getApplicationName } from './getApplicationName'; import getArchitecture from './getArchitecture'; @@ -25,6 +25,7 @@ import getDataflow from './getDataflow'; import getMitigations from './getMitigations'; import getThreats from './getThreats'; + /** * Convert threat model data into Docx format * @param data @@ -40,9 +41,11 @@ const convertToDocx = async (data: DataExchangeFormat) => { const mitigations = await getMitigations(data); const assets = await getAssets(data); + const docx = new Document({ title: data.applicationInfo?.name, creator: 'threat-composer', + description: 'This file is generated by threat-composer - a threat modeling tool: https://github.com/awslabs/threat-composer', numbering: { config: [ { @@ -51,6 +54,67 @@ const convertToDocx = async (data: DataExchangeFormat) => { }, ], }, + styles: { + default: { + title: { + paragraph: { + spacing: SPACING, + }, + }, + heading1: { + paragraph: { + spacing: SPACING, + }, + }, + heading2: { + paragraph: { + spacing: SPACING, + }, + }, + heading3: { + paragraph: { + spacing: SPACING, + }, + }, + heading4: { + paragraph: { + spacing: SPACING, + }, + }, + heading5: { + paragraph: { + spacing: SPACING, + }, + }, + heading6: { + paragraph: { + spacing: SPACING, + }, + }, + }, + paragraphStyles: [ + { + id: 'normalPara', + name: 'Normal Para', + basedOn: 'Normal', + next: 'Normal', + quickFormat: true, + paragraph: { + spacing: SPACING, + }, + }, + { + id: 'listPara', + name: 'Normal Para', + basedOn: 'Normal', + next: 'Normal', + quickFormat: true, + paragraph: { + spacing: LIST_PARA_SPACING, + }, + }, + ], + }, sections: [ { properties: { diff --git a/packages/threat-composer/src/components/report/ThreatModel/components/ThreatModelView/index.tsx b/packages/threat-composer/src/components/report/ThreatModel/components/ThreatModelView/index.tsx index 0289d887..2a6a4609 100644 --- a/packages/threat-composer/src/components/report/ThreatModel/components/ThreatModelView/index.tsx +++ b/packages/threat-composer/src/components/report/ThreatModel/components/ThreatModelView/index.tsx @@ -203,7 +203,6 @@ const ThreatModelView: FC = ({ { text: 'Download as Markdown File', id: 'markdown' }, ...(convertToDocx ? [{ text: 'Download as Word - Docx File', id: 'docx' }] : []), { text: 'Download as JSON File', id: 'json' }, - { text: 'Download as YAML File', id: 'yaml' }, ]} onItemClick={handleDownloadClick} >