d7b374f8ab
New features: - ActivityController and manager for CW sales activities (CRUD) - ForecastProductController for opportunity forecast/product lines - CW member cache with dual-layer (in-memory + Redis) resolution - Catalog category/subcategory/ecosystem taxonomy module - Quote statuses type definitions with CW mapping - User-defined fields (UDF) module with cache and event refresh - Company sites CW module with serialization - Procurement manager filters (category, ecosystem, manufacturer, price, stock) - Opportunity notes CRUD and product line management via CW API - Opportunity type definitions endpoint Updates: - OpportunityController: CW refresh, company hydration, activities, custom fields - UserController: cwIdentifier field for CW member linking - CatalogItemController: category/subcategory fields from CW - PermissionNodes: sales note/product CRUD nodes, subCategories, collectPermissions - API routes: procurement categories/filters, sales notes/products, opportunity types - Global events: UDF and member refresh intervals on startup Tests (414 passing): - ActivityController, ForecastProductController, OpportunityController unit tests - UserController cwIdentifier tests - catalogCategories, companySites, memberCache, procurement module tests - activityTypes, opportunityTypes, quoteStatuses type tests - permissionNodes subCategories and getAllPermissionNodes tests - Updated test setup with redis mock, API method mocks, and builder helpers
97 lines
2.9 KiB
TypeScript
97 lines
2.9 KiB
TypeScript
import { readFileSync } from "fs";
|
|
import { PrismaPg } from "@prisma/adapter-pg";
|
|
import { Prisma, PrismaClient } from "../generated/prisma/client";
|
|
import * as msal from "@azure/msal-node";
|
|
import { Server } from "socket.io";
|
|
import { Server as Engine } from "@socket.io/bun-engine";
|
|
import axios from "axios";
|
|
import { UnifiClient } from "./modules/unifi-api/UnifiClient";
|
|
import Redis from "ioredis";
|
|
|
|
const connectionString = `${process.env.DATABASE_URL}`;
|
|
const adapter = new PrismaPg({ connectionString });
|
|
|
|
interface EnvKey {
|
|
PORT: number;
|
|
}
|
|
|
|
// ENV CONSTANTS
|
|
|
|
export const PORT = process.env.PORT;
|
|
export const API_BASE_URL =
|
|
process.env.API_BASE_URL || `http://localhost:${PORT || 3000}`;
|
|
|
|
export const prisma = new PrismaClient({ adapter });
|
|
|
|
// Redis Client
|
|
|
|
export const redis = new Redis(process.env.REDIS_URL!);
|
|
|
|
export const sessionDuration = 30 * 24 * 60 * 60000;
|
|
export const accessTokenDuration = "10min";
|
|
export const refreshTokenDuration = "30d";
|
|
|
|
const isProduction = process.env.NODE_ENV === "production";
|
|
|
|
const readKeyFile = (path: string) => readFileSync(path).toString();
|
|
|
|
export const accessTokenPrivateKey = isProduction
|
|
? process.env.ACCESS_TOKEN_PRIVATE_KEY!
|
|
: readKeyFile(`.accessToken.key`);
|
|
export const refreshTokenPrivateKey = isProduction
|
|
? process.env.REFRESH_TOKEN_PRIVATE_KEY!
|
|
: readKeyFile(`.refreshToken.key`);
|
|
export const permissionsPrivateKey = isProduction
|
|
? process.env.PERMISSIONS_PRIVATE_KEY!
|
|
: readKeyFile(`.permissions.key`);
|
|
export const secureValuesPrivateKey = isProduction
|
|
? process.env.SECURE_VALUES_PRIVATE_KEY!
|
|
: readKeyFile(`.secureValues.key`);
|
|
export const secureValuesPublicKey = isProduction
|
|
? process.env.SECURE_VALUES_PUBLIC_KEY!
|
|
: readKeyFile(`public-keys/.secureValues.pub`);
|
|
|
|
// Microsoft Auth Constants
|
|
const msalConfig: msal.Configuration = {
|
|
auth: {
|
|
clientId: process.env.MICROSOFT_CLIENT_ID!,
|
|
authority: `https://login.microsoftonline.com/${process.env.MICROSOFT_TENANT_ID!}`,
|
|
clientSecret: process.env.MICROSOFT_CLIENT_SECRET!,
|
|
},
|
|
};
|
|
|
|
// MSAL Client Instance
|
|
export const msalClient = new msal.ConfidentialClientApplication(msalConfig);
|
|
|
|
// Socket.io
|
|
|
|
const io = new Server();
|
|
const authIO = io.of("/auth_callback");
|
|
const engine = new Engine();
|
|
|
|
io.bind(engine);
|
|
export { io, engine };
|
|
|
|
// Connectwise API Client
|
|
|
|
const connectWiseApi = axios.create({
|
|
baseURL: `https://ttscw.totaltech.net/v4_6_release/apis/3.0/`,
|
|
headers: {
|
|
Authorization: `Basic ${process.env.CW_BASIC_TOKEN}`,
|
|
clientId: `${process.env.CW_CLIENT_ID}`,
|
|
"Content-Type": "application/json",
|
|
},
|
|
});
|
|
|
|
export { connectWiseApi };
|
|
|
|
// Unifi API Constants
|
|
|
|
export const unifiControllerBaseUrl =
|
|
process.env.UNIFI_CONTROLLER_BASE_URL || "https://unifi.example.com";
|
|
export const unifiSite = process.env.UNIFI_SITE || "default";
|
|
export const unifiUsername = process.env.UNIFI_USERNAME || "admin";
|
|
export const unifiPassword = process.env.UNIFI_PASSWORD || "";
|
|
|
|
export const unifi = new UnifiClient(unifiControllerBaseUrl);
|