WIP PBS AP.
This commit is contained in:
@@ -16,6 +16,6 @@
|
|||||||
"rules": {
|
"rules": {
|
||||||
"no-console": "off"
|
"no-console": "off"
|
||||||
},
|
},
|
||||||
"settings": {},
|
"settings": {}
|
||||||
"plugins": ["cypress"]
|
//"plugins": ["cypress"]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6921,6 +6921,27 @@
|
|||||||
</translation>
|
</translation>
|
||||||
</translations>
|
</translations>
|
||||||
</concept_node>
|
</concept_node>
|
||||||
|
<concept_node>
|
||||||
|
<name>federal_tax_itc</name>
|
||||||
|
<definition_loaded>false</definition_loaded>
|
||||||
|
<description></description>
|
||||||
|
<comment></comment>
|
||||||
|
<default_text></default_text>
|
||||||
|
<translations>
|
||||||
|
<translation>
|
||||||
|
<language>en-US</language>
|
||||||
|
<approved>false</approved>
|
||||||
|
</translation>
|
||||||
|
<translation>
|
||||||
|
<language>es-MX</language>
|
||||||
|
<approved>false</approved>
|
||||||
|
</translation>
|
||||||
|
<translation>
|
||||||
|
<language>fr-CA</language>
|
||||||
|
<approved>false</approved>
|
||||||
|
</translation>
|
||||||
|
</translations>
|
||||||
|
</concept_node>
|
||||||
<concept_node>
|
<concept_node>
|
||||||
<name>gst_override</name>
|
<name>gst_override</name>
|
||||||
<definition_loaded>false</definition_loaded>
|
<definition_loaded>false</definition_loaded>
|
||||||
|
|||||||
@@ -0,0 +1,138 @@
|
|||||||
|
import { SyncOutlined } from "@ant-design/icons";
|
||||||
|
import { Button, Card, Table, Typography } from "antd";
|
||||||
|
import Dinero from "dinero.js";
|
||||||
|
import React, { useEffect, useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { connect } from "react-redux";
|
||||||
|
import { createStructuredSelector } from "reselect";
|
||||||
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
||||||
|
|
||||||
|
const mapStateToProps = createStructuredSelector({
|
||||||
|
//currentUser: selectCurrentUser
|
||||||
|
bodyshop: selectBodyshop,
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = (dispatch) => ({
|
||||||
|
//setUserLanguage: language => dispatch(setUserLanguage(language))
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(
|
||||||
|
mapStateToProps,
|
||||||
|
mapDispatchToProps
|
||||||
|
)(DmsAllocationsSummaryAp);
|
||||||
|
|
||||||
|
export function DmsAllocationsSummaryAp({ socket, bodyshop, billids, title }) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [allocationsSummary, setAllocationsSummary] = useState([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (socket.connected) {
|
||||||
|
socket.emit("pbs-calculate-allocations-ap", billids, (ack) => {
|
||||||
|
setAllocationsSummary(ack);
|
||||||
|
socket.allocationsSummary = ack;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [socket, socket.connected, billids]);
|
||||||
|
|
||||||
|
const columns = [
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.center"),
|
||||||
|
dataIndex: "center",
|
||||||
|
key: "center",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.sale"),
|
||||||
|
dataIndex: "sale",
|
||||||
|
key: "sale",
|
||||||
|
render: (text, record) => Dinero(record.sale).toFormat(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.cost"),
|
||||||
|
dataIndex: "cost",
|
||||||
|
key: "cost",
|
||||||
|
render: (text, record) => Dinero(record.cost).toFormat(),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.sale_dms_acctnumber"),
|
||||||
|
dataIndex: "sale_dms_acctnumber",
|
||||||
|
key: "sale_dms_acctnumber",
|
||||||
|
render: (text, record) =>
|
||||||
|
record.profitCenter && record.profitCenter.dms_acctnumber,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.cost_dms_acctnumber"),
|
||||||
|
dataIndex: "cost_dms_acctnumber",
|
||||||
|
key: "cost_dms_acctnumber",
|
||||||
|
render: (text, record) =>
|
||||||
|
record.costCenter && record.costCenter.dms_acctnumber,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: t("jobs.fields.dms.dms_wip_acctnumber"),
|
||||||
|
dataIndex: "dms_wip_acctnumber",
|
||||||
|
key: "dms_wip_acctnumber",
|
||||||
|
render: (text, record) =>
|
||||||
|
record.costCenter && record.costCenter.dms_wip_acctnumber,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card
|
||||||
|
title={title}
|
||||||
|
extra={
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
socket.emit("pbs-calculate-allocations-ap", billids, (ack) =>
|
||||||
|
setAllocationsSummary(ack)
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<SyncOutlined />
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Table
|
||||||
|
pagination={{ position: "top", defaultPageSize: 50 }}
|
||||||
|
columns={columns}
|
||||||
|
rowKey="center"
|
||||||
|
dataSource={allocationsSummary}
|
||||||
|
locale={{ emptyText: t("dms.labels.refreshallocations") }}
|
||||||
|
summary={() => {
|
||||||
|
const totals =
|
||||||
|
allocationsSummary &&
|
||||||
|
allocationsSummary.reduce(
|
||||||
|
(acc, val) => {
|
||||||
|
return {
|
||||||
|
totalSale: acc.totalSale.add(Dinero(val.sale)),
|
||||||
|
totalCost: acc.totalCost.add(Dinero(val.cost)),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
{
|
||||||
|
totalSale: Dinero(),
|
||||||
|
totalCost: Dinero(),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Table.Summary.Row>
|
||||||
|
<Table.Summary.Cell>
|
||||||
|
<Typography.Title level={4}>
|
||||||
|
{t("general.labels.totals")}
|
||||||
|
</Typography.Title>
|
||||||
|
</Table.Summary.Cell>
|
||||||
|
<Table.Summary.Cell>
|
||||||
|
{totals && totals.totalSale.toFormat()}
|
||||||
|
</Table.Summary.Cell>
|
||||||
|
<Table.Summary.Cell>
|
||||||
|
{
|
||||||
|
// totals.totalCost.toFormat()
|
||||||
|
}
|
||||||
|
</Table.Summary.Cell>
|
||||||
|
<Table.Summary.Cell></Table.Summary.Cell>
|
||||||
|
<Table.Summary.Cell></Table.Summary.Cell>
|
||||||
|
</Table.Summary.Row>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -89,6 +89,11 @@ function Header({
|
|||||||
{},
|
{},
|
||||||
bodyshop && bodyshop.imexshopid
|
bodyshop && bodyshop.imexshopid
|
||||||
);
|
);
|
||||||
|
const { DmsAp } = useTreatments(
|
||||||
|
["DmsAp"],
|
||||||
|
{},
|
||||||
|
bodyshop && bodyshop.imexshopid
|
||||||
|
);
|
||||||
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
@@ -264,10 +269,11 @@ function Header({
|
|||||||
{t("menus.header.accounting-receivables")}
|
{t("menus.header.accounting-receivables")}
|
||||||
</Link>
|
</Link>
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
{!(
|
{(!(
|
||||||
(bodyshop && bodyshop.cdk_dealerid) ||
|
(bodyshop && bodyshop.cdk_dealerid) ||
|
||||||
(bodyshop && bodyshop.pbs_serialnumber)
|
(bodyshop && bodyshop.pbs_serialnumber)
|
||||||
) && (
|
) ||
|
||||||
|
DmsAp.treatment === "on") && (
|
||||||
<Menu.Item key="payables">
|
<Menu.Item key="payables">
|
||||||
<Link to="/manage/accounting/payables">
|
<Link to="/manage/accounting/payables">
|
||||||
{t("menus.header.accounting-payables")}
|
{t("menus.header.accounting-payables")}
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import {
|
|||||||
import { logImEXEvent } from "../../firebase/firebase.utils";
|
import { logImEXEvent } from "../../firebase/firebase.utils";
|
||||||
import _ from "lodash";
|
import _ from "lodash";
|
||||||
import { INSERT_EXPORT_LOG } from "../../graphql/accounting.queries";
|
import { INSERT_EXPORT_LOG } from "../../graphql/accounting.queries";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop,
|
||||||
@@ -176,6 +177,13 @@ export function PayableExportAll({
|
|||||||
setLoading(false);
|
setLoading(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (bodyshop.pbs_serialnumber)
|
||||||
|
return (
|
||||||
|
<Link to={{ state: { billids }, pathname: `/manage/dmsap` }}>
|
||||||
|
<Button>{t("jobs.actions.export")}</Button>
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Button onClick={handleQbxml} loading={loading} disabled={disabled}>
|
<Button onClick={handleQbxml} loading={loading} disabled={disabled}>
|
||||||
{t("jobs.actions.exportselected")}
|
{t("jobs.actions.exportselected")}
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import {
|
|||||||
} from "../../redux/user/user.selectors";
|
} from "../../redux/user/user.selectors";
|
||||||
import { logImEXEvent } from "../../firebase/firebase.utils";
|
import { logImEXEvent } from "../../firebase/firebase.utils";
|
||||||
import { INSERT_EXPORT_LOG } from "../../graphql/accounting.queries";
|
import { INSERT_EXPORT_LOG } from "../../graphql/accounting.queries";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop,
|
||||||
@@ -179,6 +180,13 @@ export function PayableExportButton({
|
|||||||
setLoading(false);
|
setLoading(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (bodyshop.pbs_serialnumber)
|
||||||
|
return (
|
||||||
|
<Link to={{ state: { billids: [billId] }, pathname: `/manage/dmsap` }}>
|
||||||
|
<Button>{t("jobs.actions.export")}</Button>
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Button onClick={handleQbxml} loading={loading} disabled={disabled}>
|
<Button onClick={handleQbxml} loading={loading} disabled={disabled}>
|
||||||
{t("jobs.actions.export")}
|
{t("jobs.actions.export")}
|
||||||
|
|||||||
@@ -43,7 +43,11 @@ export function ShopInfoResponsibilityCenterComponent({ bodyshop, form }) {
|
|||||||
{},
|
{},
|
||||||
bodyshop && bodyshop.imexshopid
|
bodyshop && bodyshop.imexshopid
|
||||||
);
|
);
|
||||||
|
const { DmsAp } = useTreatments(
|
||||||
|
["DmsAp"],
|
||||||
|
{},
|
||||||
|
bodyshop && bodyshop.imexshopid
|
||||||
|
);
|
||||||
const [costOptions, setCostOptions] = useState(
|
const [costOptions, setCostOptions] = useState(
|
||||||
[
|
[
|
||||||
...((form.getFieldValue(["md_responsibility_centers", "costs"]) &&
|
...((form.getFieldValue(["md_responsibility_centers", "costs"]) &&
|
||||||
@@ -4159,6 +4163,121 @@ export function ShopInfoResponsibilityCenterComponent({ bodyshop, form }) {
|
|||||||
<InputNumber precision={2} />
|
<InputNumber precision={2} />
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
</LayoutFormRow>
|
</LayoutFormRow>
|
||||||
|
{DmsAp.treatment === "on" && (
|
||||||
|
<LayoutFormRow>
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenters.federal_tax_itc")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={["md_responsibility_centers", "taxes", "federal_itc", "name"]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item>
|
||||||
|
{/* <Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenter_accountnumber")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={[
|
||||||
|
"md_responsibility_centers",
|
||||||
|
"taxes",
|
||||||
|
"federal_itc",
|
||||||
|
"accountnumber",
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item> */}
|
||||||
|
{/* <Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenter_accountname")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={[
|
||||||
|
"md_responsibility_centers",
|
||||||
|
"taxes",
|
||||||
|
"federal_itc",
|
||||||
|
"accountname",
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item> */}
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenter_accountdesc")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={[
|
||||||
|
"md_responsibility_centers",
|
||||||
|
"taxes",
|
||||||
|
"federal_itc",
|
||||||
|
"accountdesc",
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item>
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenter_accountitem")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={[
|
||||||
|
"md_responsibility_centers",
|
||||||
|
"taxes",
|
||||||
|
"federal_itc",
|
||||||
|
"accountitem",
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item>
|
||||||
|
{(bodyshop.cdk_dealerid || bodyshop.pbs_serialnumber) && (
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.dms.dms_acctnumber")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={[
|
||||||
|
"md_responsibility_centers",
|
||||||
|
"taxes",
|
||||||
|
"federal_itc",
|
||||||
|
"dms_acctnumber",
|
||||||
|
]}
|
||||||
|
>
|
||||||
|
<Input />
|
||||||
|
</Form.Item>
|
||||||
|
)}
|
||||||
|
<Form.Item
|
||||||
|
label={t("bodyshop.fields.responsibilitycenter_rate")}
|
||||||
|
rules={[
|
||||||
|
{
|
||||||
|
required: true,
|
||||||
|
//message: t("general.validation.required"),
|
||||||
|
},
|
||||||
|
]}
|
||||||
|
name={["md_responsibility_centers", "taxes", "federal_itc", "rate"]}
|
||||||
|
>
|
||||||
|
<InputNumber precision={2} />
|
||||||
|
</Form.Item>
|
||||||
|
</LayoutFormRow>
|
||||||
|
)}
|
||||||
<LayoutFormRow>
|
<LayoutFormRow>
|
||||||
<Form.Item
|
<Form.Item
|
||||||
label={t("bodyshop.fields.responsibilitycenters.state_tax")}
|
label={t("bodyshop.fields.responsibilitycenters.state_tax")}
|
||||||
|
|||||||
178
client/src/pages/dms-payables/dms-payables.container.jsx
Normal file
178
client/src/pages/dms-payables/dms-payables.container.jsx
Normal file
@@ -0,0 +1,178 @@
|
|||||||
|
import { Button, Card, Col, notification, Row, Select, Space } from "antd";
|
||||||
|
import queryString from "query-string";
|
||||||
|
import React, { useEffect, useRef, useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { connect } from "react-redux";
|
||||||
|
import { useHistory, useLocation } from "react-router-dom";
|
||||||
|
import { createStructuredSelector } from "reselect";
|
||||||
|
import SocketIO from "socket.io-client";
|
||||||
|
import DmsAllocationsSummaryApComponent from "../../components/dms-allocations-summary-ap/dms-allocations-summary-ap.component";
|
||||||
|
import DmsCustomerSelector from "../../components/dms-customer-selector/dms-customer-selector.component";
|
||||||
|
import DmsLogEvents from "../../components/dms-log-events/dms-log-events.component";
|
||||||
|
import DmsPostForm from "../../components/dms-post-form/dms-post-form.component";
|
||||||
|
import { auth } from "../../firebase/firebase.utils";
|
||||||
|
import {
|
||||||
|
setBreadcrumbs,
|
||||||
|
setSelectedHeader,
|
||||||
|
} from "../../redux/application/application.actions";
|
||||||
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
||||||
|
|
||||||
|
const mapStateToProps = createStructuredSelector({
|
||||||
|
bodyshop: selectBodyshop,
|
||||||
|
});
|
||||||
|
|
||||||
|
const mapDispatchToProps = (dispatch) => ({
|
||||||
|
setBreadcrumbs: (breadcrumbs) => dispatch(setBreadcrumbs(breadcrumbs)),
|
||||||
|
setSelectedHeader: (key) => dispatch(setSelectedHeader(key)),
|
||||||
|
});
|
||||||
|
|
||||||
|
export default connect(mapStateToProps, mapDispatchToProps)(DmsContainer);
|
||||||
|
|
||||||
|
export const socket = SocketIO(
|
||||||
|
process.env.NODE_ENV === "production"
|
||||||
|
? process.env.REACT_APP_AXIOS_BASE_API_URL
|
||||||
|
: window.location.origin,
|
||||||
|
{
|
||||||
|
path: "/ws",
|
||||||
|
withCredentials: true,
|
||||||
|
auth: async (callback) => {
|
||||||
|
const token = auth.currentUser && (await auth.currentUser.getIdToken());
|
||||||
|
callback({ token });
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
export function DmsContainer({ bodyshop, setBreadcrumbs, setSelectedHeader }) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [logLevel, setLogLevel] = useState("DEBUG");
|
||||||
|
const history = useHistory();
|
||||||
|
const [logs, setLogs] = useState([]);
|
||||||
|
const search = queryString.parse(useLocation().search);
|
||||||
|
const { state } = useLocation();
|
||||||
|
|
||||||
|
const { jobId } = search;
|
||||||
|
const logsRef = useRef(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
document.title = t("titles.dms");
|
||||||
|
setSelectedHeader("dms");
|
||||||
|
setBreadcrumbs([
|
||||||
|
{
|
||||||
|
link: "/manage/accounting/receivables",
|
||||||
|
label: t("titles.bc.accounting-receivables"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
link: "/manage/dms",
|
||||||
|
label: t("titles.bc.dms"),
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
}, [t, setBreadcrumbs, setSelectedHeader]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
socket.on("connect", () => socket.emit("set-log-level", logLevel));
|
||||||
|
socket.on("reconnect", () => {
|
||||||
|
setLogs((logs) => {
|
||||||
|
return [
|
||||||
|
...logs,
|
||||||
|
{
|
||||||
|
timestamp: new Date(),
|
||||||
|
level: "WARNING",
|
||||||
|
message: "Reconnected to CDK Export Service",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
socket.on("log-event", (payload) => {
|
||||||
|
setLogs((logs) => {
|
||||||
|
return [...logs, payload];
|
||||||
|
});
|
||||||
|
});
|
||||||
|
socket.on("export-success", (payload) => {
|
||||||
|
notification.success({
|
||||||
|
message: t("jobs.successes.exported"),
|
||||||
|
});
|
||||||
|
history.push("/manage/accounting/receivables");
|
||||||
|
});
|
||||||
|
|
||||||
|
if (socket.disconnected) socket.connect();
|
||||||
|
return () => {
|
||||||
|
socket.removeAllListeners();
|
||||||
|
socket.disconnect();
|
||||||
|
};
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!state?.billids) {
|
||||||
|
history.push(`/manage/accounting/payables`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
DMS PAYABLES SCREEN
|
||||||
|
<Row gutter={[16, 16]}>
|
||||||
|
<Col md={24} lg={10}>
|
||||||
|
<DmsAllocationsSummaryApComponent
|
||||||
|
socket={socket}
|
||||||
|
billids={state?.billids}
|
||||||
|
/>
|
||||||
|
</Col>
|
||||||
|
<Col md={24} lg={14}>
|
||||||
|
{/* <DmsPostForm
|
||||||
|
socket={socket}
|
||||||
|
jobId={jobId}
|
||||||
|
// job={data && data.jobs_by_pk}
|
||||||
|
logsRef={logsRef}
|
||||||
|
/> */}
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
<DmsCustomerSelector />
|
||||||
|
|
||||||
|
<Col span={24}>
|
||||||
|
<div ref={logsRef}>
|
||||||
|
<Card
|
||||||
|
title={t("jobs.labels.dms.logs")}
|
||||||
|
extra={
|
||||||
|
<Space wrap>
|
||||||
|
<Select
|
||||||
|
placeholder="Log Level"
|
||||||
|
value={logLevel}
|
||||||
|
onChange={(value) => {
|
||||||
|
setLogLevel(value);
|
||||||
|
socket.emit("set-log-level", value);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Select.Option key="TRACE">TRACE</Select.Option>
|
||||||
|
<Select.Option key="DEBUG">DEBUG</Select.Option>
|
||||||
|
<Select.Option key="INFO">INFO</Select.Option>
|
||||||
|
<Select.Option key="WARNING">WARNING</Select.Option>
|
||||||
|
<Select.Option key="ERROR">ERROR</Select.Option>
|
||||||
|
</Select>
|
||||||
|
<Button onClick={() => setLogs([])}>Clear Logs</Button>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
setLogs([]);
|
||||||
|
socket.disconnect();
|
||||||
|
socket.connect();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Reconnect
|
||||||
|
</Button>
|
||||||
|
</Space>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<DmsLogEvents socket={socket} logs={logs} />
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export const determineDmsType = (bodyshop) => {
|
||||||
|
if (bodyshop.cdk_dealerid) return "cdk";
|
||||||
|
else {
|
||||||
|
return "pbs";
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -164,6 +164,9 @@ const EmailTest = lazy(() =>
|
|||||||
);
|
);
|
||||||
const Dashboard = lazy(() => import("../dashboard/dashboard.container"));
|
const Dashboard = lazy(() => import("../dashboard/dashboard.container"));
|
||||||
const Dms = lazy(() => import("../dms/dms.container"));
|
const Dms = lazy(() => import("../dms/dms.container"));
|
||||||
|
const DmsPayables = lazy(() =>
|
||||||
|
import("../dms-payables/dms-payables.container")
|
||||||
|
);
|
||||||
|
|
||||||
const { Content, Footer } = Layout;
|
const { Content, Footer } = Layout;
|
||||||
|
|
||||||
@@ -391,6 +394,7 @@ export function Manage({ match, conflict, bodyshop }) {
|
|||||||
<Route exact path={`${match.path}/emailtest`} component={EmailTest} />
|
<Route exact path={`${match.path}/emailtest`} component={EmailTest} />
|
||||||
<Route exact path={`${match.path}/dashboard`} component={Dashboard} />
|
<Route exact path={`${match.path}/dashboard`} component={Dashboard} />
|
||||||
<Route exact path={`${match.path}/dms`} component={Dms} />
|
<Route exact path={`${match.path}/dms`} component={Dms} />
|
||||||
|
<Route exact path={`${match.path}/dmsap`} component={DmsPayables} />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -437,6 +437,7 @@
|
|||||||
"ar": "Accounts Receivable",
|
"ar": "Accounts Receivable",
|
||||||
"ats": "ATS",
|
"ats": "ATS",
|
||||||
"federal_tax": "Federal Tax",
|
"federal_tax": "Federal Tax",
|
||||||
|
"federal_tax_itc": "Federal Tax Credit",
|
||||||
"gst_override": "GST Override Account #",
|
"gst_override": "GST Override Account #",
|
||||||
"la1": "LA1",
|
"la1": "LA1",
|
||||||
"la2": "LA2",
|
"la2": "LA2",
|
||||||
|
|||||||
@@ -437,6 +437,7 @@
|
|||||||
"ar": "",
|
"ar": "",
|
||||||
"ats": "",
|
"ats": "",
|
||||||
"federal_tax": "",
|
"federal_tax": "",
|
||||||
|
"federal_tax_itc": "",
|
||||||
"gst_override": "",
|
"gst_override": "",
|
||||||
"la1": "",
|
"la1": "",
|
||||||
"la2": "",
|
"la2": "",
|
||||||
|
|||||||
@@ -437,6 +437,7 @@
|
|||||||
"ar": "",
|
"ar": "",
|
||||||
"ats": "",
|
"ats": "",
|
||||||
"federal_tax": "",
|
"federal_tax": "",
|
||||||
|
"federal_tax_itc": "",
|
||||||
"gst_override": "",
|
"gst_override": "",
|
||||||
"la1": "",
|
"la1": "",
|
||||||
"la2": "",
|
"la2": "",
|
||||||
|
|||||||
@@ -64,11 +64,7 @@ app.use(
|
|||||||
//Email Based Paths.
|
//Email Based Paths.
|
||||||
var sendEmail = require("./server/email/sendemail.js");
|
var sendEmail = require("./server/email/sendemail.js");
|
||||||
app.post("/sendemail", fb.validateFirebaseIdToken, sendEmail.sendEmail);
|
app.post("/sendemail", fb.validateFirebaseIdToken, sendEmail.sendEmail);
|
||||||
app.post(
|
app.post("/emailbounce", bodyParser.text(), sendEmail.emailBounce);
|
||||||
"/emailbounce",
|
|
||||||
bodyParser.text(),
|
|
||||||
sendEmail.emailBounce
|
|
||||||
);
|
|
||||||
|
|
||||||
//Test route to ensure Express is responding.
|
//Test route to ensure Express is responding.
|
||||||
app.get("/test", async function (req, res) {
|
app.get("/test", async function (req, res) {
|
||||||
|
|||||||
138
server/accounting/pbs/pbs-ap-allocations.js
Normal file
138
server/accounting/pbs/pbs-ap-allocations.js
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
const path = require("path");
|
||||||
|
require("dotenv").config({
|
||||||
|
path: path.resolve(
|
||||||
|
process.cwd(),
|
||||||
|
`.env.${process.env.NODE_ENV || "development"}`
|
||||||
|
),
|
||||||
|
});
|
||||||
|
const GraphQLClient = require("graphql-request").GraphQLClient;
|
||||||
|
|
||||||
|
const queries = require("../../graphql-client/queries");
|
||||||
|
const CdkBase = require("../../web-sockets/web-socket");
|
||||||
|
const moment = require("moment");
|
||||||
|
const Dinero = require("dinero.js");
|
||||||
|
|
||||||
|
exports.default = async function (socket, billids) {
|
||||||
|
try {
|
||||||
|
CdkBase.createLogEvent(
|
||||||
|
socket,
|
||||||
|
"DEBUG",
|
||||||
|
`Received request to calculate allocations for ${billids}`
|
||||||
|
);
|
||||||
|
const { bills, bodyshops } = await QueryBillData(socket, billids);
|
||||||
|
const bodyshop = bodyshops[0];
|
||||||
|
const transactionLines = [];
|
||||||
|
|
||||||
|
bills.forEach((bill) => {
|
||||||
|
//Keep the allocations at the bill level.
|
||||||
|
const billHash = {
|
||||||
|
[bodyshop.md_responsibility_centers.taxes.federal_itc.name]: {
|
||||||
|
Account:
|
||||||
|
bodyshop.md_responsibility_centers.taxes.federal_itc.dms_acctnumber,
|
||||||
|
//ControlNumber: "String", //need to figure this out still?
|
||||||
|
Amount: Dinero(),
|
||||||
|
// Comment: "String",
|
||||||
|
//AdditionalInfo: "String",
|
||||||
|
InvoiceNumber: bill.invoice_number,
|
||||||
|
InvoiceDate: moment(bill.date).tz(bodyshop.timezone).toISOString(),
|
||||||
|
},
|
||||||
|
[bodyshop.md_responsibility_centers.taxes.state.name]: {
|
||||||
|
Account:
|
||||||
|
bodyshop.md_responsibility_centers.taxes.state.dms_acctnumber,
|
||||||
|
//ControlNumber: "String", //need to figure this out still?
|
||||||
|
Amount: Dinero(),
|
||||||
|
// Comment: "String",
|
||||||
|
//AdditionalInfo: "String",
|
||||||
|
InvoiceNumber: bill.invoice_number,
|
||||||
|
InvoiceDate: moment(bill.date).tz(bodyshop.timezone).toISOString(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
bill.billlines.forEach((bl) => {
|
||||||
|
let lineDinero = Dinero({
|
||||||
|
amount: Math.round((bl.actual_cost || 0) * 100),
|
||||||
|
})
|
||||||
|
.multiply(bl.quantity)
|
||||||
|
.multiply(bill.is_credit_memo ? -1 : 1);
|
||||||
|
const cc = getCostAccount(bl, bodyshop.md_responsibility_centers);
|
||||||
|
|
||||||
|
if (!billHash[cc.name]) {
|
||||||
|
billHash[cc.name] = {
|
||||||
|
Account: cc.dms_acctnumber,
|
||||||
|
//ControlNumber: "String", //need to figure this out still?
|
||||||
|
Amount: Dinero(),
|
||||||
|
// Comment: "String",
|
||||||
|
//AdditionalInfo: "String",
|
||||||
|
InvoiceNumber: bill.invoice_number,
|
||||||
|
InvoiceDate: moment(bill.date).tz(bodyshop.timezone).toISOString(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
//Add the line amount.
|
||||||
|
|
||||||
|
billHash[cc.name] = {
|
||||||
|
...billHash[cc.name],
|
||||||
|
Amount: billHash[cc.name].Amount.add(lineDinero),
|
||||||
|
};
|
||||||
|
|
||||||
|
//Does the line have taxes?
|
||||||
|
if (bl.applicable_taxes.federal) {
|
||||||
|
billHash[bodyshop.md_responsibility_centers.taxes.federal_itc.name] =
|
||||||
|
{
|
||||||
|
...bodyshop.md_responsibility_centers.taxes.federal_itc.name,
|
||||||
|
Amount: billHash[
|
||||||
|
bodyshop.md_responsibility_centers.taxes.federal_itc.name
|
||||||
|
].Amount.add(lineDinero.percentage(bl.federal_tax_rate || 0)),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (bl.applicable_taxes.state) {
|
||||||
|
billHash[bodyshop.md_responsibility_centers.taxes.state.name] = {
|
||||||
|
...bodyshop.md_responsibility_centers.taxes.state.name,
|
||||||
|
Amount: billHash[
|
||||||
|
bodyshop.md_responsibility_centers.taxes.state.name
|
||||||
|
].Amount.add(lineDinero.percentage(bl.state_tax_rate || 0)),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Object.keys(billHash).map((key) => {
|
||||||
|
transactionLines.push(billHash[key]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return transactionLines;
|
||||||
|
} catch (error) {
|
||||||
|
CdkBase.createLogEvent(
|
||||||
|
socket,
|
||||||
|
"ERROR",
|
||||||
|
`Error encountered in CdkCalculateAllocations. ${error}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
async function QueryBillData(socket, billids) {
|
||||||
|
CdkBase.createLogEvent(
|
||||||
|
socket,
|
||||||
|
"DEBUG",
|
||||||
|
`Querying bill data for id(s) ${billids}`
|
||||||
|
);
|
||||||
|
const client = new GraphQLClient(process.env.GRAPHQL_ENDPOINT, {});
|
||||||
|
const result = await client
|
||||||
|
.setHeaders({ Authorization: `Bearer ${socket.handshake.auth.token}` })
|
||||||
|
.request(queries.GET_PBS_AP_ALLOCATIONS, { billids: billids });
|
||||||
|
CdkBase.createLogEvent(
|
||||||
|
socket,
|
||||||
|
"TRACE",
|
||||||
|
`Bill data query result ${JSON.stringify(result, null, 2)}`
|
||||||
|
);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
//@returns the account object.
|
||||||
|
function getCostAccount(billline, respcenters) {
|
||||||
|
if (!billline.cost_center) return null;
|
||||||
|
|
||||||
|
const acctName = respcenters.defaults.costs[billline.cost_center];
|
||||||
|
|
||||||
|
return respcenters.costs.find((c) => c.name === acctName);
|
||||||
|
}
|
||||||
@@ -1633,3 +1633,45 @@ mutation ($sesid: String!, $status: String, $context: jsonb) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}`;
|
}`;
|
||||||
|
|
||||||
|
exports.GET_PBS_AP_ALLOCATIONS = `
|
||||||
|
query GET_PBS_AP_ALLOCATIONS($billids: [uuid!]) {
|
||||||
|
bodyshops(where: {associations: {active: {_eq: true}}}) {
|
||||||
|
md_responsibility_centers
|
||||||
|
timezone
|
||||||
|
}
|
||||||
|
bills(where: {id: {_in: $billids}}) {
|
||||||
|
id
|
||||||
|
date
|
||||||
|
isinhouse
|
||||||
|
invoice_number
|
||||||
|
federal_tax_rate
|
||||||
|
is_credit_memo
|
||||||
|
jobid
|
||||||
|
job {
|
||||||
|
id
|
||||||
|
ro_number
|
||||||
|
}
|
||||||
|
local_tax_rate
|
||||||
|
state_tax_rate
|
||||||
|
total
|
||||||
|
vendorid
|
||||||
|
vendor {
|
||||||
|
id
|
||||||
|
name
|
||||||
|
}
|
||||||
|
billlines {
|
||||||
|
id
|
||||||
|
actual_cost
|
||||||
|
actual_price
|
||||||
|
applicable_taxes
|
||||||
|
cost_center
|
||||||
|
deductedfromlbr
|
||||||
|
lbr_adjustment
|
||||||
|
quantity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
`;
|
||||||
|
|||||||
@@ -22,6 +22,9 @@ const {
|
|||||||
PbsSelectedCustomer,
|
PbsSelectedCustomer,
|
||||||
} = require("../accounting/pbs/pbs-job-export");
|
} = require("../accounting/pbs/pbs-job-export");
|
||||||
|
|
||||||
|
const PbsCalculateAllocationsAp =
|
||||||
|
require("../accounting/pbs/pbs-ap-allocations").default;
|
||||||
|
|
||||||
io.use(function (socket, next) {
|
io.use(function (socket, next) {
|
||||||
try {
|
try {
|
||||||
if (socket.handshake.auth.token) {
|
if (socket.handshake.auth.token) {
|
||||||
@@ -101,7 +104,7 @@ io.on("connection", (socket) => {
|
|||||||
});
|
});
|
||||||
//END CDK
|
//END CDK
|
||||||
|
|
||||||
//PBS
|
//PBS AR
|
||||||
socket.on("pbs-calculate-allocations", async (jobid, callback) => {
|
socket.on("pbs-calculate-allocations", async (jobid, callback) => {
|
||||||
const allocations = await CdkCalculateAllocations(socket, jobid);
|
const allocations = await CdkCalculateAllocations(socket, jobid);
|
||||||
createLogEvent(socket, "DEBUG", `Allocations calculated.`);
|
createLogEvent(socket, "DEBUG", `Allocations calculated.`);
|
||||||
@@ -125,7 +128,21 @@ io.on("connection", (socket) => {
|
|||||||
socket.selectedCustomerId = selectedCustomerId;
|
socket.selectedCustomerId = selectedCustomerId;
|
||||||
PbsSelectedCustomer(socket, selectedCustomerId);
|
PbsSelectedCustomer(socket, selectedCustomerId);
|
||||||
});
|
});
|
||||||
//End PBS
|
//End PBS AR
|
||||||
|
|
||||||
|
//PBS AP
|
||||||
|
socket.on("pbs-calculate-allocations-ap", async (billids, callback) => {
|
||||||
|
const allocations = await PbsCalculateAllocationsAp(socket, billids);
|
||||||
|
createLogEvent(socket, "DEBUG", `AP Allocations calculated.`);
|
||||||
|
createLogEvent(
|
||||||
|
socket,
|
||||||
|
"TRACE",
|
||||||
|
`Allocations calculated. ${JSON.stringify(allocations, null, 2)}`
|
||||||
|
);
|
||||||
|
|
||||||
|
callback(allocations);
|
||||||
|
});
|
||||||
|
//END PBS AP
|
||||||
|
|
||||||
socket.on("disconnect", () => {
|
socket.on("disconnect", () => {
|
||||||
createLogEvent(socket, "DEBUG", `User disconnected.`);
|
createLogEvent(socket, "DEBUG", `User disconnected.`);
|
||||||
|
|||||||
Reference in New Issue
Block a user