2024-10-02 15:43:20 +00:00
|
|
|
|
import React, { useContext, useEffect } from "react";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
import { InjectedRouter } from "react-router";
|
2024-10-08 19:27:55 +00:00
|
|
|
|
import { isAxiosError } from "axios";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
|
|
|
|
|
import PATHS from "router/paths";
|
2024-09-03 22:35:33 +00:00
|
|
|
|
import { LEARN_MORE_ABOUT_BASE_LINK } from "utilities/constants";
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import { getFileDetails, IFileDetails } from "utilities/file/fileUtils";
|
|
|
|
|
|
import { buildQueryStringFromParams, QueryParams } from "utilities/url";
|
|
|
|
|
|
import softwareAPI, {
|
|
|
|
|
|
MAX_FILE_SIZE_BYTES,
|
|
|
|
|
|
MAX_FILE_SIZE_MB,
|
|
|
|
|
|
} from "services/entities/software";
|
2024-09-03 22:35:33 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import { NotificationContext } from "context/notification";
|
2024-10-15 16:13:34 +00:00
|
|
|
|
import { AppContext } from "context/app";
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import { getErrorReason } from "interfaces/errors";
|
2024-09-03 22:35:33 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import CustomLink from "components/CustomLink";
|
|
|
|
|
|
import FileProgressModal from "components/FileProgressModal";
|
2024-10-15 16:13:34 +00:00
|
|
|
|
import PremiumFeatureMessage from "components/PremiumFeatureMessage";
|
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import PackageForm from "pages/SoftwarePage/components/PackageForm";
|
|
|
|
|
|
import { IPackageFormData } from "pages/SoftwarePage/components/PackageForm/PackageForm";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
import { getErrorMessage } from "./helpers";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
const baseClass = "software-custom-package";
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
interface ISoftwarePackageProps {
|
|
|
|
|
|
currentTeamId: number;
|
2024-07-18 09:20:17 +00:00
|
|
|
|
router: InjectedRouter;
|
2024-10-02 15:43:20 +00:00
|
|
|
|
isSidePanelOpen: boolean;
|
|
|
|
|
|
setSidePanelOpen: (isOpen: boolean) => void;
|
2024-07-18 09:20:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
const SoftwareCustomPackage = ({
|
|
|
|
|
|
currentTeamId,
|
2024-08-02 15:12:43 +00:00
|
|
|
|
router,
|
2024-10-02 15:43:20 +00:00
|
|
|
|
isSidePanelOpen,
|
|
|
|
|
|
setSidePanelOpen,
|
|
|
|
|
|
}: ISoftwarePackageProps) => {
|
2024-07-18 09:20:17 +00:00
|
|
|
|
const { renderFlash } = useContext(NotificationContext);
|
2024-10-15 16:13:34 +00:00
|
|
|
|
const { isPremiumTier } = useContext(AppContext);
|
2024-10-02 15:43:20 +00:00
|
|
|
|
const [uploadProgress, setUploadProgress] = React.useState(0);
|
|
|
|
|
|
const [uploadDetails, setUploadDetails] = React.useState<IFileDetails | null>(
|
|
|
|
|
|
null
|
|
|
|
|
|
);
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
const beforeUnloadHandler = (e: BeforeUnloadEvent) => {
|
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
// Next line with e.returnValue is included for legacy support
|
|
|
|
|
|
// e.g.Chrome / Edge < 119
|
|
|
|
|
|
e.returnValue = true;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// set up event listener to prevent user from leaving page while uploading
|
2024-10-02 15:43:20 +00:00
|
|
|
|
if (uploadDetails) {
|
2024-07-18 09:20:17 +00:00
|
|
|
|
addEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
removeEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// clean up event listener and timeout on component unmount
|
|
|
|
|
|
return () => {
|
|
|
|
|
|
removeEventListener("beforeunload", beforeUnloadHandler);
|
|
|
|
|
|
};
|
2024-10-02 15:43:20 +00:00
|
|
|
|
}, [uploadDetails]);
|
|
|
|
|
|
|
|
|
|
|
|
const onCancel = () => {
|
|
|
|
|
|
router.push(
|
|
|
|
|
|
`${PATHS.SOFTWARE_TITLES}?${buildQueryStringFromParams({
|
|
|
|
|
|
team_id: currentTeamId,
|
|
|
|
|
|
})}`
|
|
|
|
|
|
);
|
|
|
|
|
|
};
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
const onSubmit = async (formData: IPackageFormData) => {
|
|
|
|
|
|
console.log("submit", formData);
|
|
|
|
|
|
if (!formData.software) {
|
|
|
|
|
|
renderFlash(
|
|
|
|
|
|
"error",
|
|
|
|
|
|
`Couldn't add. Please refresh the page and try again.`
|
|
|
|
|
|
);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
|
|
|
|
|
if (formData.software && formData.software.size > MAX_FILE_SIZE_BYTES) {
|
|
|
|
|
|
renderFlash(
|
|
|
|
|
|
"error",
|
|
|
|
|
|
`Couldn't add. The maximum file size is ${MAX_FILE_SIZE_MB} MB.`
|
|
|
|
|
|
);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
setUploadDetails(getFileDetails(formData.software));
|
|
|
|
|
|
|
2024-09-17 13:40:47 +00:00
|
|
|
|
// Note: This TODO is copied to onSaveSoftwareChanges in EditSoftwareModal
|
2024-07-18 09:20:17 +00:00
|
|
|
|
// TODO: confirm we are deleting the second sentence (not modifying it) for non-self-service installers
|
|
|
|
|
|
try {
|
2024-10-02 15:43:20 +00:00
|
|
|
|
await softwareAPI.addSoftwarePackage({
|
|
|
|
|
|
data: formData,
|
|
|
|
|
|
teamId: currentTeamId,
|
|
|
|
|
|
onUploadProgress: (progressEvent) => {
|
2024-10-08 19:27:55 +00:00
|
|
|
|
const progress = progressEvent.progress || 0;
|
|
|
|
|
|
// for large uploads it seems to take a bit for the server to finalize its response so we'll keep the
|
|
|
|
|
|
// progress bar at 97% until the server response is received
|
|
|
|
|
|
setUploadProgress(Math.max(progress - 0.03, 0.01));
|
2024-10-02 15:43:20 +00:00
|
|
|
|
},
|
|
|
|
|
|
});
|
2024-07-18 09:20:17 +00:00
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
const newQueryParams: QueryParams = { team_id: currentTeamId };
|
2024-07-18 09:20:17 +00:00
|
|
|
|
if (formData.selfService) {
|
|
|
|
|
|
newQueryParams.self_service = true;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
newQueryParams.available_for_install = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
router.push(
|
|
|
|
|
|
`${PATHS.SOFTWARE_TITLES}?${buildQueryStringFromParams(newQueryParams)}`
|
|
|
|
|
|
);
|
2024-10-22 17:52:20 +00:00
|
|
|
|
|
|
|
|
|
|
renderFlash(
|
|
|
|
|
|
"success",
|
|
|
|
|
|
<>
|
|
|
|
|
|
<b>{formData.software?.name}</b> successfully added.
|
|
|
|
|
|
{formData.selfService
|
|
|
|
|
|
? " The end user can install from Fleet Desktop."
|
|
|
|
|
|
: ""}
|
|
|
|
|
|
</>
|
|
|
|
|
|
);
|
2024-07-18 09:20:17 +00:00
|
|
|
|
} catch (e) {
|
2024-10-08 19:27:55 +00:00
|
|
|
|
const isTimeout =
|
|
|
|
|
|
isAxiosError(e) &&
|
|
|
|
|
|
(e.response?.status === 504 || e.response?.status === 408);
|
2024-09-03 22:35:33 +00:00
|
|
|
|
const reason = getErrorReason(e);
|
2024-10-08 19:27:55 +00:00
|
|
|
|
|
|
|
|
|
|
if (isTimeout) {
|
|
|
|
|
|
renderFlash(
|
|
|
|
|
|
"error",
|
|
|
|
|
|
`Couldn’t upload. Request timeout. Please make sure your server and load balancer timeout is long enough.`
|
|
|
|
|
|
);
|
|
|
|
|
|
} else if (reason.includes("Fleet couldn't read the version from")) {
|
2024-09-03 22:35:33 +00:00
|
|
|
|
renderFlash(
|
|
|
|
|
|
"error",
|
2024-09-17 21:04:12 +00:00
|
|
|
|
<>
|
|
|
|
|
|
{reason}{" "}
|
2024-09-03 22:35:33 +00:00
|
|
|
|
<CustomLink
|
|
|
|
|
|
newTab
|
|
|
|
|
|
url={`${LEARN_MORE_ABOUT_BASE_LINK}/read-package-version`}
|
|
|
|
|
|
text="Learn more"
|
2024-09-17 21:04:12 +00:00
|
|
|
|
iconColor="core-fleet-white"
|
2024-09-03 22:35:33 +00:00
|
|
|
|
/>
|
2024-09-17 21:04:12 +00:00
|
|
|
|
</>
|
2024-09-03 22:35:33 +00:00
|
|
|
|
);
|
2024-09-17 21:04:12 +00:00
|
|
|
|
} else {
|
|
|
|
|
|
renderFlash("error", getErrorMessage(e));
|
2024-09-03 22:35:33 +00:00
|
|
|
|
}
|
2024-07-18 09:20:17 +00:00
|
|
|
|
}
|
2024-10-02 15:43:20 +00:00
|
|
|
|
setUploadDetails(null);
|
2024-07-18 09:20:17 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
2024-10-15 16:13:34 +00:00
|
|
|
|
if (!isPremiumTier) {
|
|
|
|
|
|
return (
|
|
|
|
|
|
<PremiumFeatureMessage className={`${baseClass}__premium-message`} />
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-07-18 09:20:17 +00:00
|
|
|
|
return (
|
|
|
|
|
|
<div className={baseClass}>
|
2024-09-17 13:40:47 +00:00
|
|
|
|
<PackageForm
|
2024-10-02 15:43:20 +00:00
|
|
|
|
showSchemaButton={!isSidePanelOpen}
|
|
|
|
|
|
onClickShowSchema={() => setSidePanelOpen(true)}
|
|
|
|
|
|
className={`${baseClass}__package-form`}
|
|
|
|
|
|
onCancel={onCancel}
|
|
|
|
|
|
onSubmit={onSubmit}
|
2024-07-18 09:20:17 +00:00
|
|
|
|
/>
|
2024-10-02 15:43:20 +00:00
|
|
|
|
{uploadDetails && (
|
|
|
|
|
|
<FileProgressModal
|
|
|
|
|
|
fileDetails={uploadDetails}
|
|
|
|
|
|
fileProgress={uploadProgress}
|
|
|
|
|
|
/>
|
|
|
|
|
|
)}
|
2024-07-18 09:20:17 +00:00
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2024-10-02 15:43:20 +00:00
|
|
|
|
export default SoftwareCustomPackage;
|