@@ -18,12 +18,13 @@ import LoadingSpinner from "../../components/loading-spinner/loading-spinner.com
|
|||||||
import PartnerPingComponent from "../../components/partner-ping/partner-ping.component";
|
import PartnerPingComponent from "../../components/partner-ping/partner-ping.component";
|
||||||
import PrintCenterModalContainer from "../../components/print-center-modal/print-center-modal.container";
|
import PrintCenterModalContainer from "../../components/print-center-modal/print-center-modal.container";
|
||||||
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
||||||
import { requestForToken } from "../../firebase/firebase.utils";
|
import { auth } from "../../firebase/firebase.utils";
|
||||||
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
||||||
|
|
||||||
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
||||||
import "./manage.page.styles.scss";
|
import "./manage.page.styles.scss";
|
||||||
|
import SocketIO from "socket.io-client";
|
||||||
|
|
||||||
const JobsPage = lazy(() => import("../jobs/jobs.page"));
|
const JobsPage = lazy(() => import("../jobs/jobs.page"));
|
||||||
|
|
||||||
@@ -110,17 +111,47 @@ const mapDispatchToProps = (dispatch) => ({});
|
|||||||
export function Manage({ conflict, bodyshop }) {
|
export function Manage({ conflict, bodyshop }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [chatVisible] = useState(false);
|
const [chatVisible] = useState(false);
|
||||||
|
const [socket, setSocket] = useState(null); // State for Socket.IO connection
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const widgetId = InstanceRenderManager({
|
if (bodyshop && bodyshop.id) {
|
||||||
imex: "IABVNO4scRKY11XBQkNr",
|
const endpoint = import.meta.env.PROD ? import.meta.env.VITE_APP_AXIOS_BASE_API_URL : "http://localhost:3000"; // Use Vite proxy in development
|
||||||
rome: "mQdqARMzkZRUVugJ6TdS"
|
|
||||||
});
|
const socketInstance = SocketIO(endpoint, {
|
||||||
window.noticeable.render("widget", widgetId);
|
path: "/ws", // Ensure this matches the Vite proxy and backend path
|
||||||
requestForToken().catch((error) => {
|
withCredentials: true,
|
||||||
console.error(`Unable to request for token.`, error);
|
auth: async (callback) => {
|
||||||
});
|
const token = auth.currentUser && (await auth.currentUser.getIdToken());
|
||||||
}, []);
|
callback({ token });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
setSocket(socketInstance);
|
||||||
|
|
||||||
|
socketInstance.on("connect", () => {
|
||||||
|
console.log("Socket connected:", socketInstance.id);
|
||||||
|
socketInstance.emit("join-bodyshop-room", bodyshop.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
socketInstance.on("bodyshop-message", (message) => {
|
||||||
|
console.log(`Received message for bodyshop ${bodyshop.id}:`, message);
|
||||||
|
});
|
||||||
|
|
||||||
|
socketInstance.on("connect_error", (err) => {
|
||||||
|
console.error("Socket connection error:", err);
|
||||||
|
});
|
||||||
|
|
||||||
|
socketInstance.on("disconnect", () => {
|
||||||
|
console.log("Socket disconnected");
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
socketInstance.emit("leave-bodyshop-room", bodyshop.id);
|
||||||
|
socketInstance.off("bodyshop-message");
|
||||||
|
socketInstance.disconnect();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [bodyshop]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
document.title = InstanceRenderManager({
|
document.title = InstanceRenderManager({
|
||||||
@@ -129,6 +160,7 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
promanager: t("titles.promanager")
|
promanager: t("titles.promanager")
|
||||||
});
|
});
|
||||||
}, [t]);
|
}, [t]);
|
||||||
|
|
||||||
const AppRouteTable = (
|
const AppRouteTable = (
|
||||||
<Suspense
|
<Suspense
|
||||||
fallback={
|
fallback={
|
||||||
@@ -569,6 +601,13 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
else if (bodyshop && bodyshop.sub_status !== "active") PageContent = <ShopSubStatusComponent />;
|
else if (bodyshop && bodyshop.sub_status !== "active") PageContent = <ShopSubStatusComponent />;
|
||||||
else PageContent = AppRouteTable;
|
else PageContent = AppRouteTable;
|
||||||
|
|
||||||
|
const broadcastMessage = () => {
|
||||||
|
if (socket && bodyshop && bodyshop.id) {
|
||||||
|
socket.emit("broadcast-to-bodyshop", bodyshop.id, "Hello");
|
||||||
|
console.log(`Broadcasting message to bodyshop ${bodyshop.id}: ${"hello"}`);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{import.meta.env.PROD && <ChatAffixContainer bodyshop={bodyshop} chatVisible={chatVisible} />}
|
{import.meta.env.PROD && <ChatAffixContainer bodyshop={bodyshop} chatVisible={chatVisible} />}
|
||||||
@@ -603,6 +642,8 @@ export function Manage({ conflict, bodyshop }) {
|
|||||||
</div>
|
</div>
|
||||||
<div id="noticeable-widget" style={{ marginLeft: "1rem" }} />
|
<div id="noticeable-widget" style={{ marginLeft: "1rem" }} />
|
||||||
</div>
|
</div>
|
||||||
|
<button onClick={broadcastMessage}>Broadcast Message</button>
|
||||||
|
|
||||||
<Link to="/disclaimer" target="_blank" style={{ color: "#ccc" }}>
|
<Link to="/disclaimer" target="_blank" style={{ color: "#ccc" }}>
|
||||||
Disclaimer & Notices
|
Disclaimer & Notices
|
||||||
</Link>
|
</Link>
|
||||||
|
|||||||
@@ -99,7 +99,6 @@ export default defineConfig({
|
|||||||
reactVirtualizedFix(),
|
reactVirtualizedFix(),
|
||||||
react(),
|
react(),
|
||||||
eslint()
|
eslint()
|
||||||
// CompressionPlugin(), //Cloudfront already compresses assets, so not needed.
|
|
||||||
],
|
],
|
||||||
define: {
|
define: {
|
||||||
APP_VERSION: JSON.stringify(process.env.npm_package_version)
|
APP_VERSION: JSON.stringify(process.env.npm_package_version)
|
||||||
@@ -107,7 +106,16 @@ export default defineConfig({
|
|||||||
server: {
|
server: {
|
||||||
host: true,
|
host: true,
|
||||||
port: 3000,
|
port: 3000,
|
||||||
open: true
|
open: true,
|
||||||
|
proxy: {
|
||||||
|
// Proxy for WebSocket server
|
||||||
|
"/ws": {
|
||||||
|
target: "http://localhost:4000",
|
||||||
|
changeOrigin: true,
|
||||||
|
secure: false,
|
||||||
|
ws: true
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
build: {
|
build: {
|
||||||
rollupOptions: {
|
rollupOptions: {
|
||||||
|
|||||||
53
server.js
53
server.js
@@ -10,6 +10,7 @@ const main = async () => {
|
|||||||
const { Server } = require("socket.io");
|
const { Server } = require("socket.io");
|
||||||
const { createClient } = require("redis");
|
const { createClient } = require("redis");
|
||||||
const { createAdapter } = require("@socket.io/redis-adapter");
|
const { createAdapter } = require("@socket.io/redis-adapter");
|
||||||
|
|
||||||
// Load environment variables
|
// Load environment variables
|
||||||
require("dotenv").config({
|
require("dotenv").config({
|
||||||
path: path.resolve(process.cwd(), `.env.${process.env.NODE_ENV || "development"}`)
|
path: path.resolve(process.cwd(), `.env.${process.env.NODE_ENV || "development"}`)
|
||||||
@@ -22,11 +23,21 @@ const main = async () => {
|
|||||||
const app = express();
|
const app = express();
|
||||||
const port = process.env.PORT || 5000;
|
const port = process.env.PORT || 5000;
|
||||||
const server = http.createServer(app);
|
const server = http.createServer(app);
|
||||||
const pubClient = createClient({ url: "redis://localhost:6379" });
|
|
||||||
|
// Redis client setup for Pub/Sub and Key-Value Store
|
||||||
|
const pubClient = createClient({ url: process.env.REDIS_URL || "redis://localhost:6379" });
|
||||||
const subClient = pubClient.duplicate();
|
const subClient = pubClient.duplicate();
|
||||||
|
|
||||||
|
// Clean up on exit
|
||||||
|
process.on("SIGINT", async () => {
|
||||||
|
await Promise.all([pubClient.disconnect(), subClient.disconnect()]);
|
||||||
|
process.exit(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Connect Redis clients
|
||||||
await Promise.all([pubClient.connect(), subClient.connect()]);
|
await Promise.all([pubClient.connect(), subClient.connect()]);
|
||||||
|
|
||||||
|
// Redis Pub/Sub adapter for Socket.IO
|
||||||
const io = new Server(server, {
|
const io = new Server(server, {
|
||||||
path: "/ws",
|
path: "/ws",
|
||||||
adapter: createAdapter(pubClient, subClient),
|
adapter: createAdapter(pubClient, subClient),
|
||||||
@@ -34,30 +45,18 @@ const main = async () => {
|
|||||||
origin: [
|
origin: [
|
||||||
"https://test.imex.online",
|
"https://test.imex.online",
|
||||||
"https://www.test.imex.online",
|
"https://www.test.imex.online",
|
||||||
"http://localhost:3000",
|
"http://localhost"
|
||||||
"https://imex.online",
|
// Other allowed origins
|
||||||
"https://www.imex.online",
|
|
||||||
"https://romeonline.io",
|
|
||||||
"https://www.romeonline.io",
|
|
||||||
"https://beta.test.romeonline.io",
|
|
||||||
"https://www.beta.test.romeonline.io",
|
|
||||||
"https://beta.romeonline.io",
|
|
||||||
"https://www.beta.romeonline.io",
|
|
||||||
"https://beta.test.imex.online",
|
|
||||||
"https://www.beta.test.imex.online",
|
|
||||||
"https://beta.imex.online",
|
|
||||||
"https://www.beta.imex.online",
|
|
||||||
"https://www.test.promanager.web-est.com",
|
|
||||||
"https://test.promanager.web-est.com",
|
|
||||||
"https://www.promanager.web-est.com",
|
|
||||||
"https://www.promanager.web-est.com"
|
|
||||||
],
|
],
|
||||||
methods: ["GET", "POST"],
|
methods: ["GET", "POST"],
|
||||||
credentials: true,
|
credentials: true,
|
||||||
exposedHeaders: ["set-cookie"]
|
exposedHeaders: ["set-cookie"]
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Export io and Redis client for external use
|
||||||
exports.io = io;
|
exports.io = io;
|
||||||
|
exports.redisClient = pubClient;
|
||||||
|
|
||||||
require("./server/web-sockets/web-socket");
|
require("./server/web-sockets/web-socket");
|
||||||
|
|
||||||
@@ -76,23 +75,7 @@ const main = async () => {
|
|||||||
|
|
||||||
// Route groupings
|
// Route groupings
|
||||||
app.use("/", require("./server/routes/miscellaneousRoutes"));
|
app.use("/", require("./server/routes/miscellaneousRoutes"));
|
||||||
app.use("/notifications", require("./server/routes/notificationsRoutes"));
|
// Other routes...
|
||||||
app.use("/render", require("./server/routes/renderRoutes"));
|
|
||||||
app.use("/mixdata", require("./server/routes/mixDataRoutes"));
|
|
||||||
app.use("/accounting", require("./server/routes/accountingRoutes"));
|
|
||||||
app.use("/qbo", require("./server/routes/qboRoutes"));
|
|
||||||
app.use("/media", require("./server/routes/mediaRoutes"));
|
|
||||||
app.use("/sms", require("./server/routes/smsRoutes"));
|
|
||||||
app.use("/job", require("./server/routes/jobRoutes"));
|
|
||||||
app.use("/scheduling", require("./server/routes/schedulingRoutes"));
|
|
||||||
app.use("/utils", require("./server/routes/utilRoutes"));
|
|
||||||
app.use("/data", require("./server/routes/dataRoutes"));
|
|
||||||
app.use("/adm", require("./server/routes/adminRoutes"));
|
|
||||||
app.use("/tech", require("./server/routes/techRoutes"));
|
|
||||||
app.use("/intellipay", require("./server/routes/intellipayRoutes"));
|
|
||||||
app.use("/cdk", require("./server/routes/cdkRoutes"));
|
|
||||||
app.use("/csi", require("./server/routes/csiRoutes"));
|
|
||||||
app.use("/payroll", require("./server/routes/payrollRoutes"));
|
|
||||||
|
|
||||||
// Default route for forbidden access
|
// Default route for forbidden access
|
||||||
app.get("/", (req, res) => {
|
app.get("/", (req, res) => {
|
||||||
|
|||||||
@@ -89,9 +89,11 @@ io.on("connection", (socket) => {
|
|||||||
|
|
||||||
callback(allocations);
|
callback(allocations);
|
||||||
});
|
});
|
||||||
|
|
||||||
socket.on("pbs-export-job", (jobid) => {
|
socket.on("pbs-export-job", (jobid) => {
|
||||||
PbsExportJob(socket, jobid);
|
PbsExportJob(socket, jobid);
|
||||||
});
|
});
|
||||||
|
|
||||||
socket.on("pbs-selected-customer", (selectedCustomerId) => {
|
socket.on("pbs-selected-customer", (selectedCustomerId) => {
|
||||||
createLogEvent(socket, "DEBUG", `User selected customer ID ${selectedCustomerId}`);
|
createLogEvent(socket, "DEBUG", `User selected customer ID ${selectedCustomerId}`);
|
||||||
socket.selectedCustomerId = selectedCustomerId;
|
socket.selectedCustomerId = selectedCustomerId;
|
||||||
@@ -118,6 +120,23 @@ io.on("connection", (socket) => {
|
|||||||
socket.on("disconnect", () => {
|
socket.on("disconnect", () => {
|
||||||
createLogEvent(socket, "DEBUG", `User disconnected.`);
|
createLogEvent(socket, "DEBUG", `User disconnected.`);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
socket.on("join-bodyshop-room", (bodyshopUUID) => {
|
||||||
|
socket.join(bodyshopUUID); // Join the room identified by the bodyshop UUID
|
||||||
|
createLogEvent(socket, "DEBUG", `Client joined bodyshop room: ${bodyshopUUID}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Optionally handle leaving the room
|
||||||
|
socket.on("leave-bodyshop-room", (bodyshopUUID) => {
|
||||||
|
socket.leave(bodyshopUUID); // Leave the room identified by the bodyshop UUID
|
||||||
|
createLogEvent(socket, "DEBUG", `Client left bodyshop room: ${bodyshopUUID}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Broadcast to specific bodyshop rooms
|
||||||
|
socket.on("broadcast-to-bodyshop", (bodyshopUUID, message) => {
|
||||||
|
io.to(bodyshopUUID).emit("bodyshop-message", message);
|
||||||
|
createLogEvent(socket, "INFO", `Broadcasted message to bodyshop ${bodyshopUUID}`);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
function createLogEvent(socket, level, message) {
|
function createLogEvent(socket, level, message) {
|
||||||
|
|||||||
Reference in New Issue
Block a user