kepler.gl
Version:
kepler.gl is a webgl based application to visualize large scale location data in the browser
331 lines (324 loc) • 43.3 kB
JavaScript
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.CSV_NULLS = exports.COLUMN_TYPES_PRIORITIES = void 0;
exports.columnTypeToFieldType = columnTypeToFieldType;
exports.fieldTypeToColumnType = fieldTypeToColumnType;
exports.processCsvRowObject = processCsvRowObject;
exports.processGeojson = processGeojson;
exports.processKeplerglJSONforDuckDb = processKeplerglJSONforDuckDb;
var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator"));
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator"));
var _geojsonNormalize = _interopRequireDefault(require("@mapbox/geojson-normalize"));
var _commonUtils = require("@kepler.gl/common-utils");
var _constants = require("@kepler.gl/constants");
var _processors = require("@kepler.gl/processors");
var _utils = require("@kepler.gl/utils");
function _createForOfIteratorHelper(r, e) { var t = "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (!t) { if (Array.isArray(r) || (t = _unsupportedIterableToArray(r)) || e && r && "number" == typeof r.length) { t && (r = t); var _n = 0, F = function F() {}; return { s: F, n: function n() { return _n >= r.length ? { done: !0 } : { done: !1, value: r[_n++] }; }, e: function e(r) { throw r; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var o, a = !0, u = !1; return { s: function s() { t = t.call(r); }, n: function n() { var r = t.next(); return a = r.done, r; }, e: function e(r) { u = !0, o = r; }, f: function f() { try { a || null == t["return"] || t["return"](); } finally { if (u) throw o; } } }; }
function _unsupportedIterableToArray(r, a) { if (r) { if ("string" == typeof r) return _arrayLikeToArray(r, a); var t = {}.toString.call(r).slice(8, -1); return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0; } }
function _arrayLikeToArray(r, a) { (null == a || a > r.length) && (a = r.length); for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e]; return n; }
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2["default"])(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } // SPDX-License-Identifier: MIT
// Copyright contributors to the kepler.gl project
var CSV_NULLS = exports.CSV_NULLS = /^(null|NULL|Null|NaN|\/N||)$/;
var COLUMN_TYPES = /*#__PURE__*/function (COLUMN_TYPES) {
COLUMN_TYPES["GEOMETRY"] = "GEOMETRY";
COLUMN_TYPES["BOOLEAN"] = "BOOLEAN";
COLUMN_TYPES["INTEGER"] = "INTEGER";
COLUMN_TYPES["FLOAT"] = "FLOAT";
COLUMN_TYPES["TIME"] = "TIME";
COLUMN_TYPES["DATE"] = "DATE";
COLUMN_TYPES["TIMESTAMP"] = "TIMESTAMP";
COLUMN_TYPES["VARCHAR"] = "VARCHAR";
return COLUMN_TYPES;
}(COLUMN_TYPES || {}); // https://duckdb.org/docs/data/csv/auto_detection.html#type-detection
var COLUMN_TYPES_PRIORITIES = exports.COLUMN_TYPES_PRIORITIES = ['GEOMETRY', 'BOOLEAN', 'INTEGER', 'FLOAT', 'TIME', 'DATE', 'TIMESTAMP', 'VARCHAR'];
function columnTypeToFieldType(columnType) {
switch (columnType) {
case COLUMN_TYPES.BOOLEAN:
return _constants.ALL_FIELD_TYPES["boolean"];
case COLUMN_TYPES.DATE:
return _constants.ALL_FIELD_TYPES.date;
case COLUMN_TYPES.INTEGER:
return _constants.ALL_FIELD_TYPES.integer;
case COLUMN_TYPES.FLOAT:
return _constants.ALL_FIELD_TYPES.real;
case COLUMN_TYPES.VARCHAR:
return _constants.ALL_FIELD_TYPES.string;
case COLUMN_TYPES.TIMESTAMP:
return _constants.ALL_FIELD_TYPES.timestamp;
case COLUMN_TYPES.GEOMETRY:
return _constants.ALL_FIELD_TYPES.geojson;
default:
// is there any other type we didn't cover?
console.warn("Unsupported DuckDB Column type: ".concat(columnType));
return null;
}
}
function fieldTypeToColumnType(fieldType) {
switch (fieldType) {
case _constants.ALL_FIELD_TYPES["boolean"]:
return COLUMN_TYPES.BOOLEAN;
case _constants.ALL_FIELD_TYPES.date:
return COLUMN_TYPES.DATE;
case _constants.ALL_FIELD_TYPES.integer:
return COLUMN_TYPES.INTEGER;
case _constants.ALL_FIELD_TYPES.real:
return COLUMN_TYPES.FLOAT;
case _constants.ALL_FIELD_TYPES.string:
return COLUMN_TYPES.VARCHAR;
case _constants.ALL_FIELD_TYPES.timestamp:
return COLUMN_TYPES.TIMESTAMP;
case _constants.ALL_FIELD_TYPES.geojson:
return COLUMN_TYPES.GEOMETRY;
default:
// is there any other type we didn't cover?
console.warn("Unsupported Field type for DuckDb: ".concat(fieldType));
return null;
}
}
/*
* Process uploaded exported Keplergl json map.
* We need to populate field with proper DuckDb compatible type.
*/
// TODO: merge with logic from processCsvRowObject. Different formats: [[]] vs [{}]
function processKeplerglJSONforDuckDb(_x) {
return _processKeplerglJSONforDuckDb.apply(this, arguments);
}
/*
* Process uploaded csv returned by loaders.gl as row object and string value
*/
function _processKeplerglJSONforDuckDb() {
_processKeplerglJSONforDuckDb = (0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee(keplerJson) {
var res, datasets, _iterator, _step, _loop2;
return _regenerator["default"].wrap(function _callee$(_context2) {
while (1) switch (_context2.prev = _context2.next) {
case 0:
res = (0, _processors.processKeplerglJSON)(keplerJson);
datasets = (res === null || res === void 0 ? void 0 : res.datasets) || [];
_iterator = _createForOfIteratorHelper(datasets);
_context2.prev = 3;
_loop2 = /*#__PURE__*/_regenerator["default"].mark(function _loop2() {
var dataset, rowsAll, fieldsAll, header, sample, schema, fieldsUpd;
return _regenerator["default"].wrap(function _loop2$(_context) {
while (1) switch (_context.prev = _context.next) {
case 0:
dataset = _step.value;
rowsAll = dataset.data.rows;
fieldsAll = dataset.data.fields;
header = fieldsAll.map(function (f) {
return f.name;
});
sample = (0, _commonUtils.getSampleForTypeAnalyze)({
fields: header,
rows: rowsAll
});
if (!(sample.length > 0)) {
_context.next = 11;
break;
}
_context.next = 8;
return sniffCsvSchema(sample);
case 8:
_context.t0 = _context.sent;
_context.next = 12;
break;
case 11:
_context.t0 = fieldsAll.reduce(function (acc, field) {
acc[field.name] = fieldTypeToColumnType(field.type);
return acc;
}, {});
case 12:
schema = _context.t0;
fieldsUpd = consolidateFieldTypes(dataset.data.fields, schema);
dataset.data.fields = fieldsAll.map(function (f, i) {
return _objectSpread(_objectSpread({}, fieldsUpd[i]), f);
});
case 15:
case "end":
return _context.stop();
}
}, _loop2);
});
_iterator.s();
case 6:
if ((_step = _iterator.n()).done) {
_context2.next = 10;
break;
}
return _context2.delegateYield(_loop2(), "t0", 8);
case 8:
_context2.next = 6;
break;
case 10:
_context2.next = 15;
break;
case 12:
_context2.prev = 12;
_context2.t1 = _context2["catch"](3);
_iterator.e(_context2.t1);
case 15:
_context2.prev = 15;
_iterator.f();
return _context2.finish(15);
case 18:
return _context2.abrupt("return", res);
case 19:
case "end":
return _context2.stop();
}
}, _callee, null, [[3, 12, 15, 18]]);
}));
return _processKeplerglJSONforDuckDb.apply(this, arguments);
}
function processCsvRowObject(_x2) {
return _processCsvRowObject.apply(this, arguments);
}
function _processCsvRowObject() {
_processCsvRowObject = (0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee2(rawData) {
var header, sample, schema, fieldsWAnalyzerType, fields;
return _regenerator["default"].wrap(function _callee2$(_context3) {
while (1) switch (_context3.prev = _context3.next) {
case 0:
if (!(!Array.isArray(rawData) || !rawData.length)) {
_context3.next = 2;
break;
}
return _context3.abrupt("return", {
fields: [],
rows: []
});
case 2:
header = Object.keys(rawData[0]); // [lat, lng, value]
// row object can still contain values like `Null` or `N/A`
cleanUpFalsyCsvValue(rawData);
sample = (0, _commonUtils.getSampleForTypeAnalyze)({
fields: header,
rows: rawData
}); // add sample data to duckdb and get schema
_context3.next = 7;
return sniffCsvSchema(sample);
case 7:
schema = _context3.sent;
fieldsWAnalyzerType = (0, _commonUtils.getFieldsFromData)(sample, header);
fields = consolidateFieldTypes(fieldsWAnalyzerType, schema);
return _context3.abrupt("return", {
rows: rawData,
fields: fields
});
case 11:
case "end":
return _context3.stop();
}
}, _callee2);
}));
return _processCsvRowObject.apply(this, arguments);
}
function cleanUpFalsyCsvValue(rows) {
var re = new RegExp(CSV_NULLS, 'g');
var _loop = function _loop(i) {
Object.keys(rows[i]).forEach(function (key) {
// here we parse empty data as null
if (typeof rows[i][key] === 'string' && rows[i][key].match(re)) {
rows[i][key] = null;
}
});
};
for (var i = 0; i < rows.length; i++) {
_loop(i);
}
}
// align type analyzer types with DuckDB csv auto detected column types
function consolidateFieldTypes(fields, schema) {
return fields.map(function (field) {
var columnName = field.name;
var detectedColumnType = schema[columnName];
// TODO columnTypeToFieldType tranforms detected timestamps to string,
// completely breaking time filter logic.
// const fieldType = columnTypeToFieldType(detectedColumnType) || field.type;
return _objectSpread(_objectSpread({}, field), {}, {
// type: fieldType,
duckDBColumnType: detectedColumnType
});
});
}
function toCSVRow(row) {
return "".concat(row.map(function (r) {
var rToStr = (0, _commonUtils.notNullorUndefined)(r) ? String(r).replace(/"/g, '\\"') : '';
return rToStr.includes(',') ? "\"".concat(rToStr, "\"") : rToStr;
}).join(','), "\n");
}
// Use DucckDB to detect csv column schema
function sniffCsvSchema(_x3) {
return _sniffCsvSchema.apply(this, arguments);
}
function _sniffCsvSchema() {
_sniffCsvSchema = (0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee3(sample) {
var headerRow, csvContent, db, c, fileName, result, schemaResult, columns, schema, i;
return _regenerator["default"].wrap(function _callee3$(_context4) {
while (1) switch (_context4.prev = _context4.next) {
case 0:
if (!(!Array.isArray(sample) || sample.length < 0)) {
_context4.next = 2;
break;
}
return _context4.abrupt("return");
case 2:
headerRow = toCSVRow(Object.keys(sample[0]));
csvContent = sample.reduce(function (accu, row) {
return "".concat(accu).concat(toCSVRow(Object.values(row)));
}, headerRow);
db = (0, _utils.getApplicationConfig)().database;
if (db) {
_context4.next = 8;
break;
}
console.error('The database is not configured properly.');
return _context4.abrupt("return");
case 8:
_context4.next = 10;
return db.connect();
case 10:
c = _context4.sent;
fileName = (0, _commonUtils.generateHashId)();
_context4.next = 14;
return db.registerFileText("".concat(fileName, "-").concat(_commonUtils.generateHashId, ".csv"), csvContent);
case 14:
_context4.next = 16;
return c.query("\n FROM sniff_csv('".concat(fileName, "-").concat(_commonUtils.generateHashId, ".csv', \n sample_size = 500,\n auto_type_candidates = ['FLOAT', 'INTEGER', 'TIMESTAMP', 'DATE', 'TIME', 'VARCHAR', 'BOOLEAN']);\n "));
case 16:
result = _context4.sent;
schemaResult = result.toArray().map(function (row) {
return row.toJSON();
});
columns = schemaResult[0].Columns;
schema = {};
for (i = 0; i < columns.length; i++) {
schema[columns.get(i).name] = columns.get(i).type;
}
return _context4.abrupt("return", schema);
case 22:
case "end":
return _context4.stop();
}
}, _callee3);
}));
return _sniffCsvSchema.apply(this, arguments);
}
function processGeojson(rawData) {
var normalizedGeojson = (0, _geojsonNormalize["default"])(rawData);
if (!normalizedGeojson || !Array.isArray(normalizedGeojson.features)) {
var error = new Error("Read File Failed: File is not a valid GeoJSON. Read more about [supported file format](".concat(_constants.GUIDES_FILE_FORMAT_DOC, ")"));
throw error;
}
// @ts-expect-error Don't pass empty fields, as duck db outputs an empty dataset
return {
rows: normalizedGeojson
// TODO get fields to preserve field names?
// fields: []
};
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_geojsonNormalize","_interopRequireDefault","require","_commonUtils","_constants","_processors","_utils","_createForOfIteratorHelper","r","e","t","Symbol","iterator","Array","isArray","_unsupportedIterableToArray","length","_n","F","s","n","done","value","f","TypeError","o","a","u","call","next","_arrayLikeToArray","toString","slice","constructor","name","from","test","ownKeys","Object","keys","getOwnPropertySymbols","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","defineProperty","CSV_NULLS","exports","COLUMN_TYPES","COLUMN_TYPES_PRIORITIES","columnTypeToFieldType","columnType","BOOLEAN","ALL_FIELD_TYPES","DATE","date","INTEGER","integer","FLOAT","real","VARCHAR","string","TIMESTAMP","timestamp","GEOMETRY","geojson","console","warn","concat","fieldTypeToColumnType","fieldType","processKeplerglJSONforDuckDb","_x","_processKeplerglJSONforDuckDb","_asyncToGenerator2","_regenerator","mark","_callee","keplerJson","res","datasets","_iterator","_step","_loop2","wrap","_callee$","_context2","prev","processKeplerglJSON","dataset","rowsAll","fieldsAll","header","sample","schema","fieldsUpd","_loop2$","_context","data","rows","fields","map","getSampleForTypeAnalyze","sniffCsvSchema","t0","sent","reduce","acc","field","type","consolidateFieldTypes","i","stop","delegateYield","t1","finish","abrupt","processCsvRowObject","_x2","_processCsvRowObject","_callee2","rawData","fieldsWAnalyzerType","_callee2$","_context3","cleanUpFalsyCsvValue","getFieldsFromData","re","RegExp","_loop","key","match","columnName","detectedColumnType","duckDBColumnType","toCSVRow","row","rToStr","notNullorUndefined","String","replace","includes","join","_x3","_sniffCsvSchema","_callee3","headerRow","csvContent","db","c","fileName","result","schemaResult","columns","_callee3$","_context4","accu","values","getApplicationConfig","database","error","connect","generateHashId","registerFileText","query","toArray","toJSON","Columns","get","processGeojson","normalizedGeojson","normalize","features","Error","GUIDES_FILE_FORMAT_DOC"],"sources":["../../src/processors/data-processor.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport * as arrow from 'apache-arrow';\nimport normalize from '@mapbox/geojson-normalize';\n\nimport {\n  getSampleForTypeAnalyze,\n  getFieldsFromData,\n  generateHashId,\n  notNullorUndefined\n} from '@kepler.gl/common-utils';\nimport {ProtoDatasetField} from '@kepler.gl/types';\nimport {ALL_FIELD_TYPES, GUIDES_FILE_FORMAT_DOC} from '@kepler.gl/constants';\nimport {processKeplerglJSON} from '@kepler.gl/processors';\nimport {getApplicationConfig} from '@kepler.gl/utils';\n\nexport const CSV_NULLS = /^(null|NULL|Null|NaN|\\/N||)$/;\n\ntype RowsAsArray = any[][];\ntype RowsAsObject = Record<string, unknown>[];\ntype RowData = RowsAsArray | RowsAsObject;\n\nexport type ProcessorResult = {\n  cols?: arrow.Vector[];\n  rows: RowData;\n  fields: ProtoDatasetField[];\n};\n\nenum COLUMN_TYPES {\n  GEOMETRY = 'GEOMETRY',\n  BOOLEAN = 'BOOLEAN',\n  INTEGER = 'INTEGER',\n  FLOAT = 'FLOAT',\n  TIME = 'TIME',\n  DATE = 'DATE',\n  TIMESTAMP = 'TIMESTAMP',\n  VARCHAR = 'VARCHAR'\n}\n// https://duckdb.org/docs/data/csv/auto_detection.html#type-detection\nexport const COLUMN_TYPES_PRIORITIES = [\n  'GEOMETRY',\n  'BOOLEAN',\n  'INTEGER',\n  'FLOAT',\n  'TIME',\n  'DATE',\n  'TIMESTAMP',\n  'VARCHAR'\n];\n\nexport function columnTypeToFieldType(columnType: string): string | null {\n  switch (columnType) {\n    case COLUMN_TYPES.BOOLEAN:\n      return ALL_FIELD_TYPES.boolean;\n    case COLUMN_TYPES.DATE:\n      return ALL_FIELD_TYPES.date;\n    case COLUMN_TYPES.INTEGER:\n      return ALL_FIELD_TYPES.integer;\n    case COLUMN_TYPES.FLOAT:\n      return ALL_FIELD_TYPES.real;\n    case COLUMN_TYPES.VARCHAR:\n      return ALL_FIELD_TYPES.string;\n    case COLUMN_TYPES.TIMESTAMP:\n      return ALL_FIELD_TYPES.timestamp;\n    case COLUMN_TYPES.GEOMETRY:\n      return ALL_FIELD_TYPES.geojson;\n    default:\n      // is there any other type we didn't cover?\n      console.warn(`Unsupported DuckDB Column type: ${columnType}`);\n      return null;\n  }\n}\n\nexport function fieldTypeToColumnType(fieldType: string): string | null {\n  switch (fieldType) {\n    case ALL_FIELD_TYPES.boolean:\n      return COLUMN_TYPES.BOOLEAN;\n    case ALL_FIELD_TYPES.date:\n      return COLUMN_TYPES.DATE;\n    case ALL_FIELD_TYPES.integer:\n      return COLUMN_TYPES.INTEGER;\n    case ALL_FIELD_TYPES.real:\n      return COLUMN_TYPES.FLOAT;\n    case ALL_FIELD_TYPES.string:\n      return COLUMN_TYPES.VARCHAR;\n    case ALL_FIELD_TYPES.timestamp:\n      return COLUMN_TYPES.TIMESTAMP;\n    case ALL_FIELD_TYPES.geojson:\n      return COLUMN_TYPES.GEOMETRY;\n    default:\n      // is there any other type we didn't cover?\n      console.warn(`Unsupported Field type for DuckDb: ${fieldType}`);\n      return null;\n  }\n}\n\n/*\n * Process uploaded exported Keplergl json map.\n * We need to populate field with proper DuckDb compatible type.\n */\n// TODO: merge with logic from processCsvRowObject. Different formats: [[]] vs [{}]\nexport async function processKeplerglJSONforDuckDb(\n  keplerJson\n): Promise<ReturnType<typeof processKeplerglJSON>> {\n  const res = processKeplerglJSON(keplerJson);\n\n  const datasets = res?.datasets || [];\n  for (const dataset of datasets) {\n    const rowsAll = dataset.data.rows;\n    const fieldsAll = dataset.data.fields;\n\n    const header = fieldsAll.map(f => f.name);\n    const sample = getSampleForTypeAnalyze({fields: header, rows: rowsAll});\n\n    const schema =\n      sample.length > 0\n        ? await sniffCsvSchema(sample)\n        : fieldsAll.reduce((acc, field) => {\n            acc[field.name] = fieldTypeToColumnType(field.type);\n            return acc;\n          }, {});\n    const fieldsUpd = consolidateFieldTypes(dataset.data.fields, schema);\n\n    dataset.data.fields = fieldsAll.map((f, i) => {\n      return {...fieldsUpd[i], ...f};\n    });\n  }\n\n  return res;\n}\n\n/*\n * Process uploaded csv returned by loaders.gl as row object and string value\n */\nexport async function processCsvRowObject(\n  rawData: Record<string, string | null>[]\n): Promise<ProcessorResult> {\n  if (!Array.isArray(rawData) || !rawData.length) {\n    return {\n      fields: [],\n      rows: []\n    };\n  }\n\n  const header = Object.keys(rawData[0]); // [lat, lng, value]\n\n  // row object can still contain values like `Null` or `N/A`\n  cleanUpFalsyCsvValue(rawData);\n  const sample = getSampleForTypeAnalyze({fields: header, rows: rawData});\n  // add sample data to duckdb and get schema\n  const schema = await sniffCsvSchema(sample);\n\n  const fieldsWAnalyzerType = getFieldsFromData(sample, header);\n\n  const fields = consolidateFieldTypes(fieldsWAnalyzerType, schema);\n\n  return {rows: rawData, fields};\n}\n\nfunction cleanUpFalsyCsvValue(rows: RowData): void {\n  const re = new RegExp(CSV_NULLS, 'g');\n  for (let i = 0; i < rows.length; i++) {\n    Object.keys(rows[i]).forEach(key => {\n      // here we parse empty data as null\n      if (typeof rows[i][key] === 'string' && (rows[i][key] as string).match(re)) {\n        rows[i][key] = null;\n      }\n    });\n  }\n}\n\n// align type analyzer types with DuckDB csv auto detected column types\nfunction consolidateFieldTypes(fields, schema) {\n  return fields.map(field => {\n    const columnName = field.name;\n\n    const detectedColumnType = schema[columnName];\n\n    // TODO columnTypeToFieldType tranforms detected timestamps to string,\n    // completely breaking time filter logic.\n    // const fieldType = columnTypeToFieldType(detectedColumnType) || field.type;\n\n    return {\n      ...field,\n      // type: fieldType,\n      duckDBColumnType: detectedColumnType\n    };\n  });\n}\n\nfunction toCSVRow(row: unknown[]): string {\n  return `${row\n    .map(r => {\n      const rToStr = notNullorUndefined(r) ? String(r).replace(/\"/g, '\\\\\"') : '';\n      return rToStr.includes(',') ? `\"${rToStr}\"` : rToStr;\n    })\n    .join(',')}\\n`;\n}\n\n// Use DucckDB to detect csv column schema\nasync function sniffCsvSchema(sample: RowData) {\n  if (!Array.isArray(sample) || sample.length < 0) {\n    return;\n  }\n  const headerRow = toCSVRow(Object.keys(sample[0]));\n\n  const csvContent = (sample as (any[] | Record<string, unknown>)[]).reduce(\n    (accu: string, row) => `${accu}${toCSVRow(Object.values(row))}`,\n    headerRow\n  );\n\n  const db = getApplicationConfig().database;\n  if (!db) {\n    console.error('The database is not configured properly.');\n    return;\n  }\n  const c = await db.connect();\n\n  const fileName = generateHashId();\n  await db.registerFileText(`${fileName}-${generateHashId}.csv`, csvContent);\n\n  // https://duckdb.org/docs/data/csv/auto_detection.html\n  const result = await c.query(`\n        FROM sniff_csv('${fileName}-${generateHashId}.csv', \n        sample_size = 500,\n        auto_type_candidates = ['FLOAT', 'INTEGER', 'TIMESTAMP', 'DATE', 'TIME', 'VARCHAR', 'BOOLEAN']);\n    `);\n\n  const schemaResult = result.toArray().map(row => row.toJSON());\n\n  const columns = schemaResult[0].Columns;\n  const schema = {};\n  for (let i = 0; i < columns.length; i++) {\n    schema[columns.get(i).name] = columns.get(i).type;\n  }\n\n  return schema;\n}\n\nexport function processGeojson(rawData: unknown): ProcessorResult {\n  const normalizedGeojson = normalize(rawData);\n\n  if (!normalizedGeojson || !Array.isArray(normalizedGeojson.features)) {\n    const error = new Error(\n      `Read File Failed: File is not a valid GeoJSON. Read more about [supported file format](${GUIDES_FILE_FORMAT_DOC})`\n    );\n    throw error;\n  }\n\n  // @ts-expect-error Don't pass empty fields, as duck db outputs an empty dataset\n  return {\n    rows: normalizedGeojson\n    // TODO get fields to preserve field names?\n    // fields: []\n  };\n}\n"],"mappings":"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","ignoreList":[]}
;