Browse Source

feat: add export excel on public views

pull/2851/head
Louis Delbosc 2 years ago
parent
commit
f4cca91d13
  1. 75
      packages/nc-gui/components/project/spreadsheet/components/MoreActions.vue
  2. 86
      packages/nocodb/src/lib/meta/api/dataApis/helpers.ts
  3. 109
      packages/nocodb/src/lib/meta/api/publicApis/publicDataExportApis.ts

75
packages/nc-gui/components/project/spreadsheet/components/MoreActions.vue

@ -233,27 +233,62 @@ export default {
async exportExcel() { async exportExcel() {
let offset = 0; let offset = 0;
let c = 1; let c = 1;
const res = await this.$api.dbViewRow.export( try {
'noco', while (!isNaN(offset) && offset > -1) {
this.projectName, let res;
this.meta.title, if (this.publicViewId) {
this.selectedView.title, console.log('IF', this.publicViewId)
ExportTypes.EXCEL, res = await this.$api.public.csvExport(this.publicViewId, ExportTypes.EXCEL, {
{ responseType: 'blob',
responseType: 'base64', query: {
query: { fields:
offset, this.queryParams &&
}, this.queryParams.fieldsOrder &&
} this.queryParams.fieldsOrder.filter(c => this.queryParams.showFields[c]),
); offset,
const workbook = XLSX.read(res.data, { type: 'base64' }); sortArrJson: JSON.stringify(
XLSX.writeFile(workbook, `${this.meta.title}_exported_${c++}.xlsx`); this.reqPayload &&
this.reqPayload.sorts &&
this.reqPayload.sorts.map(({ fk_column_id, direction }) => ({
direction,
fk_column_id,
}))
),
filterArrJson: JSON.stringify(this.reqPayload && this.reqPayload.filters),
},
headers: {
'xc-password': this.reqPayload && this.reqPayload.password,
},
});
} else {
console.log('ELSE')
res = await this.$api.dbViewRow.export(
'noco',
this.projectName,
this.meta.title,
this.selectedView.title,
ExportTypes.EXCEL,
{
responseType: 'base64',
query: {
offset,
},
}
);
}
const workbook = XLSX.read(res.data, { type: 'base64' });
XLSX.writeFile(workbook, `${this.meta.title}_exported_${c++}.xlsx`);
offset = +res.headers['nc-export-offset']; offset = +res.headers['nc-export-offset'];
if (offset > -1) { if (offset > -1) {
this.$toast.info('Downloading more files').goAway(3000); this.$toast.info('Downloading more files').goAway(3000);
} else { } else {
this.$toast.success('Successfully exported all table data').goAway(3000); this.$toast.success('Successfully exported all table data').goAway(3000);
}
}
} catch (e) {
console.log(e);
this.$toast.error(e.message).goAway(3000);
} }
}, },
async exportCsv() { async exportCsv() {

86
packages/nocodb/src/lib/meta/api/dataApis/helpers.ts

@ -57,53 +57,10 @@ export async function extractXlsxData(view: View, req: Request) {
dbDriver: NcConnectionMgrv2.get(base), dbDriver: NcConnectionMgrv2.get(base),
}); });
let offset = +req.query.offset || 0; const { offset, dbRows, elapsed } = await getDbRows(baseModel, view, req);
const limit = 100; const data = XLSX.utils.json_to_sheet(dbRows);
// const size = +process.env.NC_EXPORT_MAX_SIZE || 1024;
const timeout = +process.env.NC_EXPORT_MAX_TIMEOUT || 5000;
const csvRows = [];
const startTime = process.hrtime();
let elapsed, temp;
for (
elapsed = 0;
elapsed < timeout;
offset += limit,
temp = process.hrtime(startTime),
elapsed = temp[0] * 1000 + temp[1] / 1000000
) {
const rows = await nocoExecute(
await getAst({
query: req.query,
includePkByDefault: false,
model: view.model,
view,
}),
await baseModel.list({ ...req.query, offset, limit }),
{},
req.query
);
if (!rows?.length) {
offset = -1;
break;
}
for (const row of rows) {
const csvRow = { ...row };
for (const column of view.model.columns) { return { offset, dbRows, elapsed, data };
if (isSystemColumn(column) && !view.show_system_fields) continue;
csvRow[column.title] = await serializeCellValue({
value: row[column.title],
column,
});
}
csvRows.push(csvRow);
}
}
const data = XLSX.utils.json_to_sheet(csvRows);
return { offset, csvRows, elapsed, data };
} }
export async function extractCsvData(view: View, req: Request) { export async function extractCsvData(view: View, req: Request) {
@ -126,11 +83,27 @@ export async function extractCsvData(view: View, req: Request) {
dbDriver: NcConnectionMgrv2.get(base), dbDriver: NcConnectionMgrv2.get(base),
}); });
const { offset, dbRows, elapsed } = await getDbRows(baseModel, view, req);
const data = papaparse.unparse(
{
fields: view.model.columns.map((c) => c.title),
data: dbRows,
},
{
escapeFormulae: true,
}
);
return { offset, dbRows, elapsed, data };
}
async function getDbRows(baseModel, view: View, req: Request) {
let offset = +req.query.offset || 0; let offset = +req.query.offset || 0;
const limit = 100; const limit = 100;
// const size = +process.env.NC_EXPORT_MAX_SIZE || 1024; // const size = +process.env.NC_EXPORT_MAX_SIZE || 1024;
const timeout = +process.env.NC_EXPORT_MAX_TIMEOUT || 5000; const timeout = +process.env.NC_EXPORT_MAX_TIMEOUT || 5000;
const csvRows = []; const dbRows = [];
const startTime = process.hrtime(); const startTime = process.hrtime();
let elapsed, temp; let elapsed, temp;
@ -159,30 +132,19 @@ export async function extractCsvData(view: View, req: Request) {
} }
for (const row of rows) { for (const row of rows) {
const csvRow = { ...row }; const dbRow = { ...row };
for (const column of view.model.columns) { for (const column of view.model.columns) {
if (isSystemColumn(column) && !view.show_system_fields) continue; if (isSystemColumn(column) && !view.show_system_fields) continue;
csvRow[column.title] = await serializeCellValue({ dbRow[column.title] = await serializeCellValue({
value: row[column.title], value: row[column.title],
column, column,
}); });
} }
csvRows.push(csvRow); dbRows.push(dbRow);
} }
} }
return { offset, dbRows, elapsed };
const data = papaparse.unparse(
{
fields: view.model.columns.map((c) => c.title),
data: csvRows,
},
{
escapeFormulae: true,
}
);
return { offset, csvRows, elapsed, data };
} }
export async function serializeCellValue({ export async function serializeCellValue({

109
packages/nocodb/src/lib/meta/api/publicApis/publicDataExportApis.ts

@ -1,4 +1,5 @@
import { Request, Response, Router } from 'express'; import { Request, Response, Router } from 'express';
import * as XLSX from 'xlsx';
import View from '../../../models/View'; import View from '../../../models/View';
import Model from '../../../models/Model'; import Model from '../../../models/Model';
import Base from '../../../models/Base'; import Base from '../../../models/Base';
@ -12,8 +13,38 @@ import LookupColumn from '../../../models/LookupColumn';
import catchError, { NcError } from '../../helpers/catchError'; import catchError, { NcError } from '../../helpers/catchError';
import getAst from '../../../db/sql-data-mapper/lib/sql/helpers/getAst'; import getAst from '../../../db/sql-data-mapper/lib/sql/helpers/getAst';
async function exportExcel(req: Request, res: Response) {
const view = await View.getByUUID(req.params.publicDataUuid);
if (!view) NcError.notFound('Not found');
if (view.type !== ViewTypes.GRID) NcError.notFound('Not found');
if (view.password && view.password !== req.headers?.['xc-password']) {
NcError.forbidden(ErrorMessages.INVALID_SHARED_VIEW_PASSWORD);
}
const model = await view.getModelWithInfo();
await view.getColumns();
const { offset, dbRows, elapsed } = await getDbRows(model, view, req);
const data = XLSX.utils.json_to_sheet(dbRows);
const wb = XLSX.utils.book_new();
XLSX.utils.book_append_sheet(wb, data, view.title);
const buf = XLSX.write(wb, { type: "base64", bookType: "xlsx" });
res.set({
'Access-Control-Expose-Headers': 'nc-export-offset',
'nc-export-offset': offset,
'nc-export-elapsed-time': elapsed,
'Content-Disposition': `attachment; filename="${encodeURI(
view.title
)}-export.xlsx"`,
});
res.end(buf);
}
async function exportCsv(req: Request, res: Response) { async function exportCsv(req: Request, res: Response) {
const view = await View.getByUUID(req.params.publicDataUuid); const view = await View.getByUUID(req.params.publicDataUuid);
const fields = req.query.fields;
if (!view) NcError.notFound('Not found'); if (!view) NcError.notFound('Not found');
if (view.type !== ViewTypes.GRID) NcError.notFound('Not found'); if (view.type !== ViewTypes.GRID) NcError.notFound('Not found');
@ -25,6 +56,40 @@ async function exportCsv(req: Request, res: Response) {
const model = await view.getModelWithInfo(); const model = await view.getModelWithInfo();
await view.getColumns(); await view.getColumns();
const { offset, dbRows, elapsed } = await getDbRows(model, view, req);
const data = papaparse.unparse(
{
fields: model.columns
.sort((c1, c2) =>
Array.isArray(fields)
? fields.indexOf(c1.title as any) - fields.indexOf(c2.title as any)
: 0
)
.filter(
(c) =>
!fields || !Array.isArray(fields) || fields.includes(c.title as any)
)
.map((c) => c.title),
data: dbRows,
},
{
escapeFormulae: true,
}
);
res.set({
'Access-Control-Expose-Headers': 'nc-export-offset',
'nc-export-offset': offset,
'nc-export-elapsed-time': elapsed,
'Content-Disposition': `attachment; filename="${encodeURI(
view.title
)}-export.csv"`,
});
res.send(data);
}
async function getDbRows(model, view: View, req: Request) {
view.model.columns = view.columns view.model.columns = view.columns
.filter((c) => c.show) .filter((c) => c.show)
.map( .map(
@ -35,7 +100,6 @@ async function exportCsv(req: Request, res: Response) {
if (!model) NcError.notFound('Table not found'); if (!model) NcError.notFound('Table not found');
const fields = req.query.fields;
const listArgs: any = { ...req.query }; const listArgs: any = { ...req.query };
try { try {
listArgs.filterArr = JSON.parse(listArgs.filterArrJson); listArgs.filterArr = JSON.parse(listArgs.filterArrJson);
@ -62,7 +126,7 @@ async function exportCsv(req: Request, res: Response) {
const limit = 100; const limit = 100;
// const size = +process.env.NC_EXPORT_MAX_SIZE || 1024; // const size = +process.env.NC_EXPORT_MAX_SIZE || 1024;
const timeout = +process.env.NC_EXPORT_MAX_TIMEOUT || 5000; const timeout = +process.env.NC_EXPORT_MAX_TIMEOUT || 5000;
const csvRows = []; const dbRows = [];
const startTime = process.hrtime(); const startTime = process.hrtime();
let elapsed, temp; let elapsed, temp;
@ -86,47 +150,18 @@ async function exportCsv(req: Request, res: Response) {
} }
for (const row of rows) { for (const row of rows) {
const csvRow = { ...row }; const dbRow = { ...row };
for (const column of view.model.columns) { for (const column of view.model.columns) {
csvRow[column.title] = await serializeCellValue({ dbRow[column.title] = await serializeCellValue({
value: row[column.title], value: row[column.title],
column, column,
}); });
} }
csvRows.push(csvRow); dbRows.push(dbRow);
} }
} }
return { offset, dbRows, elapsed };
const data = papaparse.unparse(
{
fields: model.columns
.sort((c1, c2) =>
Array.isArray(fields)
? fields.indexOf(c1.title as any) - fields.indexOf(c2.title as any)
: 0
)
.filter(
(c) =>
!fields || !Array.isArray(fields) || fields.includes(c.title as any)
)
.map((c) => c.title),
data: csvRows,
},
{
escapeFormulae: true,
}
);
res.set({
'Access-Control-Expose-Headers': 'nc-export-offset',
'nc-export-offset': offset,
'nc-export-elapsed-time': elapsed,
'Content-Disposition': `attachment; filename="${encodeURI(
view.title
)}-export.csv"`,
});
res.send(data);
} }
async function serializeCellValue({ async function serializeCellValue({
@ -198,4 +233,8 @@ router.get(
'/api/v1/db/public/shared-view/:publicDataUuid/rows/export/csv', '/api/v1/db/public/shared-view/:publicDataUuid/rows/export/csv',
catchError(exportCsv) catchError(exportCsv)
); );
router.get(
'/api/v1/db/public/shared-view/:publicDataUuid/rows/export/excel',
catchError(exportExcel)
);
export default router; export default router;

Loading…
Cancel
Save