@lineai/municipal-intel
Version:
AI-first municipal data API providing natural language descriptions of building permits and planning applications from major US cities
287 lines • 20.2 kB
JavaScript
;
/**
* @lineai/municipal-intel
*
* Access municipal planning applications, building permits, and construction
* activity data from major US cities.
*/
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __exportStar = (this && this.__exportStar) || function(m, exports) {
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.createMunicipalIntel = exports.MunicipalIntel = void 0;
// Types
__exportStar(require("./types"), exports);
// Clients
__exportStar(require("./clients"), exports);
// Registry
__exportStar(require("./registry"), exports);
// Main API class
const clients_1 = require("./clients");
const registry_1 = require("./registry");
/**
* Main municipal intelligence API
*/
class MunicipalIntel {
constructor(config = {}) {
this.clientFactory = new clients_1.ClientFactory(config);
this.registry = registry_1.sourceRegistry;
}
/**
* Search municipal projects
*/
async search(params) {
let source;
if (params.municipalityId) {
// Search specific municipality
const foundSource = this.registry.getSource(params.municipalityId);
if (!foundSource) {
throw new Error(`Municipality not found: ${params.municipalityId}`);
}
source = foundSource;
}
else {
// Default to first available source if no municipality specified
const sources = this.registry.getImplementationReadySources();
if (sources.length === 0) {
throw new Error('No available sources for search');
}
source = sources[0];
}
const client = this.clientFactory.createClient(source, params);
return client.search();
}
/**
* Get a project by ID from a specific source
*/
async getProject(sourceId, projectId) {
const source = this.registry.getSource(sourceId);
if (!source) {
throw new Error(`Source not found: ${sourceId}`);
}
// Create minimal params for getProject - we only need the municipalityId
const params = { municipalityId: sourceId };
const client = this.clientFactory.createClient(source, params);
return client.getProject(projectId);
}
/**
* Get a project by its URL
*/
async getByUrl(url) {
const sourceId = this.extractSourceFromUrl(url);
if (!sourceId) {
throw new Error(`Cannot determine source from URL: ${url}`);
}
const source = this.registry.getSource(sourceId);
if (!source) {
throw new Error(`Source not found: ${sourceId}`);
}
const params = { municipalityId: sourceId };
const client = this.clientFactory.createClient(source, params);
return client.getByUrl(url);
}
/**
* Extract source ID from municipal-intel URL
*/
extractSourceFromUrl(url) {
try {
const urlObj = new URL(url);
// Expected format: /projects/{sourceId}/{datasetId}/{projectId}
const pathParts = urlObj.pathname.split('/');
if (pathParts.length >= 4 && pathParts[1] === 'projects') {
return pathParts[2]; // Return the sourceId part
}
return null;
}
catch (error) {
return null;
}
}
/**
* Get available municipalities with their datasets (AI Discovery API)
*/
getAvailableMunicipalities() {
const sources = this.registry.getAllSources();
return sources.map(source => {
var _a;
return ({
id: source.id,
name: source.name,
state: source.state,
datasets: ((_a = source.api) === null || _a === void 0 ? void 0 : _a.datasets)
? Object.entries(source.api.datasets).map(([id, dataset]) => ({
id,
name: dataset.name
}))
: []
});
});
}
/**
* Get search capabilities for a municipality
*/
getSearchCapabilities(municipalityId) {
var _a, _b, _c;
const source = this.registry.getSource(municipalityId);
if (!source) {
throw new Error(`Municipality not found: ${municipalityId}`);
}
// For now, return capabilities based on what our Socrata client supports
// This could be enhanced to be source-specific
const supportedFilters = ['submitDateFrom', 'submitDateTo', 'statuses', 'addresses', 'keywords'];
const supportedSorts = ['submitDate', 'address'];
const limitations = [];
// Check if value field is available for value filters
if ((_a = source.api) === null || _a === void 0 ? void 0 : _a.datasets) {
const primaryDataset = Object.values(source.api.datasets)[0];
if ((_b = primaryDataset === null || primaryDataset === void 0 ? void 0 : primaryDataset.fieldMappings) === null || _b === void 0 ? void 0 : _b.value) {
supportedFilters.push('minValue', 'maxValue');
supportedSorts.push('value');
}
else {
limitations.push('No value field available - minValue/maxValue filters not supported');
}
// Check if approval date is available
if ((_c = primaryDataset === null || primaryDataset === void 0 ? void 0 : primaryDataset.fieldMappings) === null || _c === void 0 ? void 0 : _c.approvalDate) {
supportedFilters.push('approvalDateFrom', 'approvalDateTo');
supportedSorts.push('approvalDate');
}
}
return {
supportedFilters,
supportedSorts,
limitations: limitations.length > 0 ? limitations : undefined
};
}
/**
* Get field schema for a dataset
*/
getDatasetSchema(municipalityId, datasetId) {
var _a;
const source = this.registry.getSource(municipalityId);
if (!source || !((_a = source.api) === null || _a === void 0 ? void 0 : _a.datasets)) {
throw new Error(`Municipality or datasets not found: ${municipalityId}`);
}
// Get the specified dataset or the first one
const dataset = datasetId
? source.api.datasets[datasetId]
: Object.values(source.api.datasets)[0];
if (!dataset) {
throw new Error(`Dataset not found: ${datasetId || 'default'}`);
}
// Convert fields to schema format
const fieldMappings = dataset.fieldMappings || {};
const searchableLogicalFields = Object.keys(fieldMappings);
return dataset.fields.map(fieldName => {
const isSearchable = Object.values(fieldMappings).includes(fieldName);
// Determine field type based on field name patterns
let type = 'string';
if (fieldName.includes('date') || fieldName.includes('_date')) {
type = 'date';
}
else if (fieldName.includes('cost') || fieldName.includes('value') || fieldName.includes('amount')) {
type = 'number';
}
return {
name: fieldName,
type,
searchable: isSearchable,
description: isSearchable
? `Searchable field mapped to: ${searchableLogicalFields.find(logical => fieldMappings[logical] === fieldName)}`
: undefined
};
});
}
/**
* Get available sources
*/
getSources(filters) {
let sources = this.registry.getAllSources();
if (filters === null || filters === void 0 ? void 0 : filters.state) {
const filterState = filters.state.toLowerCase();
sources = sources.filter(s => s.state.toLowerCase() === filterState);
}
if (filters === null || filters === void 0 ? void 0 : filters.type) {
sources = sources.filter(s => s.type === filters.type);
}
if (filters === null || filters === void 0 ? void 0 : filters.priority) {
sources = sources.filter(s => s.priority === filters.priority);
}
if ((filters === null || filters === void 0 ? void 0 : filters.enabled) !== undefined) {
sources = sources.filter(s => (s.enabled !== false) === filters.enabled);
}
return sources;
}
// /**
// * Check health of a specific source
// */
// async healthCheck(sourceId: string) {
// const source = this.registry.getSource(sourceId);
// if (!source) {
// throw new Error(`Source not found: ${sourceId}`);
// }
//
// const client = this.clientFactory.createClient(source);
// const health = await client.healthCheck();
//
// // Update registry with health info
// this.registry.updateSourceStatus(sourceId, {
// lastChecked: health.lastChecked.toISOString(),
// lastError: health.status === 'unhealthy' ? health.error : undefined
// });
//
// return health;
// }
/**
* Set universal Socrata authentication token
*/
setSocrataToken(token) {
this.clientFactory.setSocrataToken(token);
}
/**
* Register a new source at runtime
*/
registerSource(source) {
this.registry.registerSource(source);
}
/**
* Unregister a runtime source
*/
unregisterSource(id) {
return this.registry.unregisterSource(id);
}
/**
* Check if a source is built-in or runtime-added
*/
isBuiltInSource(id) {
return this.registry.isBuiltInSource(id);
}
/**
* Get registry information
*/
getRegistryInfo() {
return this.registry.getRegistryInfo();
}
}
exports.MunicipalIntel = MunicipalIntel;
/**
* Create a new MunicipalIntel instance
*/
function createMunicipalIntel(config) {
return new MunicipalIntel(config);
}
exports.createMunicipalIntel = createMunicipalIntel;
// Default export
exports.default = MunicipalIntel;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"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"}