import React, { useEffect, useContext } from "react";
import { makeStyles } from "@mui/styles";
import ReactGA from 'react-ga4';
import SecurityFramework from '../components/SecurityFramework.jsx';
import theme from '../theme.jsx';
import {
Badge,
Avatar,
Grid,
Paper,
Tooltip,
Divider,
Button,
TextField,
FormControl,
IconButton,
Menu,
MenuItem,
FormControlLabel,
Chip,
Switch,
Typography,
Zoom,
CircularProgress,
Dialog,
DialogTitle,
DialogActions,
DialogContent,
} from "@mui/material";
import {
Check as CheckIcon,
GridOn as GridOnIcon,
List as ListIcon,
Close as CloseIcon,
Compare as CompareIcon,
Maximize as MaximizeIcon,
Minimize as MinimizeIcon,
AddCircle as AddCircleIcon,
Toc as TocIcon,
Send as SendIcon,
Search as SearchIcon,
FileCopy as FileCopyIcon,
Delete as DeleteIcon,
BubbleChart as BubbleChartIcon,
Restore as RestoreIcon,
Cached as CachedIcon,
GetApp as GetAppIcon,
Apps as AppsIcon,
Edit as EditIcon,
MoreVert as MoreVertIcon,
PlayArrow as PlayArrowIcon,
Add as AddIcon,
Publish as PublishIcon,
CloudUpload as CloudUploadIcon,
CloudDownload as CloudDownloadIcon,
} from "@mui/icons-material";
import { DataGrid, GridToolbar } from "@mui/x-data-grid";
//import JSONPretty from 'react-json-pretty';
//import JSONPrettyMon from 'react-json-pretty/dist/monikai'
import Dropzone from "../components/Dropzone.jsx";
import { useNavigate, Link, useParams } from "react-router-dom";
//import { useAlert
import { ToastContainer, toast } from "react-toastify"
import { MuiChipsInput } from "mui-chips-input";
import { v4 as uuidv4 } from "uuid";
const inputColor = "#383B40";
const surfaceColor = "#27292D";
const svgSize = 24;
const imagesize = 22;
const useStyles = makeStyles((theme) => ({
datagrid: {
border: 0,
"& .MuiDataGrid-columnsContainer": {
backgroundColor:
theme.palette.type === "light" ? "#fafafa" : theme.palette.inputColor,
},
"& .MuiDataGrid-iconSeparator": {
display: "none",
},
"& .MuiDataGrid-colCell, .MuiDataGrid-cell": {
borderRight: `1px solid ${
theme.palette.type === "light" ? "white" : "#303030"
}`,
},
"& .MuiDataGrid-columnsContainer, .MuiDataGrid-cell": {
borderBottom: `1px solid ${
theme.palette.type === "light" ? "#f0f0f0" : "#303030"
}`,
},
"& .MuiDataGrid-cell": {
color:
theme.palette.type === "light" ? "white" : "rgba(255,255,255,0.65)",
},
"& .MuiPaginationItem-root, .MuiTablePagination-actions, .MuiTablePagination-caption":
{
borderRadius: 0,
color: "white",
},
},
}));
const chipStyle = {
backgroundColor: "#3d3f43",
marginRight: 5,
paddingLeft: 5,
paddingRight: 5,
height: 28,
cursor: "pointer",
borderColor: "#3d3f43",
color: "white",
};
const GettingStarted = (props) => {
const { globalUrl, isLoggedIn, isLoaded, userdata } = props;
document.title = "Getting Started with Shuffle";
//const alert = useAlert();
const classes = useStyles(theme);
let navigate = useNavigate();
const imgSize = 60;
const referenceUrl = globalUrl + "/api/v1/hooks/";
var upload = "";
const [workflows, setWorkflows] = React.useState([]);
const [filteredWorkflows, setFilteredWorkflows] = React.useState([]);
const [selectedWorkflow, setSelectedWorkflow] = React.useState({});
const [workflowDone, setWorkflowDone] = React.useState(false);
const [selectedWorkflowId, setSelectedWorkflowId] = React.useState("");
const [field1, setField1] = React.useState("");
const [field2, setField2] = React.useState("");
const [downloadUrl, setDownloadUrl] = React.useState(
"https://github.com/frikky/shuffle-workflows"
);
const [videoViewOpen, setVideoViewOpen] = React.useState(false)
const [downloadBranch, setDownloadBranch] = React.useState("master");
const [loadWorkflowsModalOpen, setLoadWorkflowsModalOpen] = React.useState(false);
const [exportModalOpen, setExportModalOpen] = React.useState(false);
const [exportData, setExportData] = React.useState("");
const [modalOpen, setModalOpen] = React.useState(false);
const [newWorkflowName, setNewWorkflowName] = React.useState("");
const [newWorkflowDescription, setNewWorkflowDescription] =
React.useState("");
const [newWorkflowTags, setNewWorkflowTags] = React.useState([]);
const [defaultReturnValue, setDefaultReturnValue] = React.useState("");
const [deleteModalOpen, setDeleteModalOpen] = React.useState(false);
const [publishModalOpen, setPublishModalOpen] = React.useState(false);
const [editingWorkflow, setEditingWorkflow] = React.useState({});
const [importLoading, setImportLoading] = React.useState(false);
const [isDropzone, setIsDropzone] = React.useState(false);
const [view, setView] = React.useState("grid");
const [filters, setFilters] = React.useState([]);
const [submitLoading, setSubmitLoading] = React.useState(false);
const [actionImageList, setActionImageList] = React.useState([]);
const [firstLoad, setFirstLoad] = React.useState(true);
const isCloud =
window.location.host === "localhost:3002" ||
window.location.host === "shuffler.io";
const findWorkflow = (filters) => {
if (filters.length === 0) {
setFilteredWorkflows(workflows);
return;
}
var newWorkflows = [];
for (var workflowKey in workflows) {
const curWorkflow = workflows[workflowKey];
var found = [false];
if (curWorkflow.tags === undefined || curWorkflow.tags === null) {
found = filters.map((filter) =>
curWorkflow.name.toLowerCase().includes(filter)
);
} else {
found = filters.map((filter) => {
const newfilter = filter.toLowerCase();
if (filter === undefined) {
return false;
}
if (curWorkflow.name.toLowerCase().includes(filter.toLowerCase())) {
return true;
} else if (curWorkflow.tags.includes(filter)) {
return true;
} else if (curWorkflow.owner === filter) {
return true;
} else if (curWorkflow.org_id === filter) {
return true;
} else if (
curWorkflow.actions !== null &&
curWorkflow.actions !== undefined
) {
for (var key in curWorkflow.actions) {
const action = curWorkflow.actions[key];
if (
action.app_name.toLowerCase() === newfilter ||
action.app_name.toLowerCase().includes(newfilter)
) {
return true;
}
}
}
return false;
});
}
if (found.every((v) => v === true)) {
newWorkflows.push(curWorkflow);
continue;
}
}
if (newWorkflows.length !== workflows.length) {
setFilteredWorkflows(newWorkflows);
}
};
const addFilter = (data) => {
if (data === null || data === undefined) {
return;
}
if (data.includes("<") && data.includes(">")) {
return;
}
if (filters.includes(data) || filters.includes(data.toLowerCase())) {
return;
}
filters.push(data.toLowerCase());
setFilters(filters);
findWorkflow(filters);
};
const removeFilter = (index) => {
var newfilters = filters;
if (index < 0) {
console.log("Can't handle index: ", index);
return;
}
newfilters.splice(index, 1);
if (newfilters.length === 0) {
newfilters = [];
setFilters(newfilters);
} else {
setFilters(newfilters);
}
findWorkflow(newfilters);
};
const exportVerifyModal = exportModalOpen ? (
) : null;
const publishModal = publishModalOpen ? (
) : null;
const deleteModal = deleteModalOpen ? (
) : null;
const uploadFile = (e) => {
const isDropzone =
e.dataTransfer === undefined ? false : e.dataTransfer.files.length > 0;
const files = isDropzone ? e.dataTransfer.files : e.target.files;
const reader = new FileReader();
toast("Starting upload. Please wait while we validate the workflows");
try {
reader.addEventListener("load", (e) => {
var data = e.target.result;
setIsDropzone(false);
try {
data = JSON.parse(reader.result);
} catch (e) {
toast("Invalid JSON: " + e);
return;
}
// Initialize the workflow itself
setNewWorkflow(
data.name,
data.description,
data.tags,
data.default_return_value,
{},
false
)
.then((response) => {
if (response !== undefined) {
// SET THE FULL THING
data.id = response.id;
// Actually create it
setNewWorkflow(
data.name,
data.description,
data.tags,
data.default_return_value,
data,
false
).then((response) => {
if (response !== undefined) {
toast(`Successfully imported ${data.name}`);
}
});
}
})
.catch((error) => {
toast("Import error: " + error.toString());
});
});
} catch (e) {
console.log("Error in dropzone: ", e);
}
reader.readAsText(files[0]);
};
useEffect(() => {
if (isDropzone) {
setIsDropzone(false);
}
}, [isDropzone]);
const getAvailableWorkflows = () => {
fetch(globalUrl + "/api/v1/workflows", {
method: "GET",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
credentials: "include",
})
.then((response) => {
if (response.status !== 200) {
console.log("Status not 200 for workflows :O!: ", response.status);
if (isCloud) {
window.location.pathname = "/login";
}
toast("Failed getting workflows.");
setWorkflowDone(true);
return;
}
return response.json();
})
.then((responseJson) => {
if (responseJson !== undefined) {
setWorkflows(responseJson);
if (responseJson !== undefined) {
var actionnamelist = [];
var parsedactionlist = [];
for (var key in responseJson) {
for (var actionkey in responseJson[key].actions) {
const action = responseJson[key].actions[actionkey];
//console.log("Action: ", action)
if (actionnamelist.includes(action.app_name)) {
continue;
}
actionnamelist.push(action.app_name);
parsedactionlist.push(action);
}
}
//console.log(parsedactionlist)
setActionImageList(parsedactionlist);
}
setFilteredWorkflows(responseJson);
setWorkflowDone(true);
// Ensures the zooming happens only once per load
setTimeout(() => {
setFirstLoad(false)
setVideoViewOpen(true)
}, 100)
} else {
if (isLoggedIn) {
toast("An error occurred while loading workflows");
}
return;
}
})
.catch((error) => {
setVideoViewOpen(true)
toast(error.toString());
});
};
// eslint-disable-next-line react-hooks/exhaustive-deps
useEffect(() => {
if (workflows.length <= 0) {
const tmpView = localStorage.getItem("view");
if (tmpView !== undefined && tmpView !== null) {
setView(tmpView);
}
//setFirstrequest(false);
getAvailableWorkflows();
}
}, [])
const viewStyle = {
color: "#ffffff",
width: "100%",
display: "flex",
minWidth: 1024,
maxWidth: 1024,
margin: "auto",
};
const emptyWorkflowStyle = {
paddingTop: "200px",
width: 1024,
margin: "auto",
};
const boxStyle = {
padding: "20px 20px 20px 20px",
width: "100%",
height: "250px",
color: "white",
backgroundColor: surfaceColor,
display: "flex",
flexDirection: "column",
};
const paperAppContainer = {
display: "flex",
flexWrap: "wrap",
alignContent: "space-between",
marginTop: 5,
};
const paperAppStyle = {
minHeight: 130,
maxHeight: 130,
overflow: "hidden",
width: "100%",
color: "white",
backgroundColor: surfaceColor,
padding: "12px 12px 0px 15px",
borderRadius: 5,
display: "flex",
boxSizing: "border-box",
position: "relative",
};
const gridContainer = {
height: "auto",
color: "white",
margin: "10px",
backgroundColor: surfaceColor,
};
const workflowActionStyle = {
display: "flex",
width: 160,
height: 44,
justifyContent: "space-between",
};
const exportAllWorkflows = () => {
for (var key in workflows) {
exportWorkflow(workflows[key], false);
}
};
const deduplicateIds = (data) => {
if (data.triggers !== null && data.triggers !== undefined) {
for (var key in data.triggers) {
const trigger = data.triggers[key];
if (trigger.app_name === "Shuffle Workflow") {
if (trigger.parameters.length > 2) {
trigger.parameters[2].value = "";
}
}
if (trigger.status === "running") {
trigger.status = "stopped";
}
const newId = uuidv4();
if (trigger.trigger_type === "WEBHOOK") {
if (
trigger.parameters !== undefined &&
trigger.parameters !== null &&
trigger.parameters.length === 2
) {
trigger.parameters[0].value =
referenceUrl + "webhook_" + trigger.id;
trigger.parameters[1].value = "webhook_" + trigger.id;
} else if (
trigger.parameters !== undefined &&
trigger.parameters !== null &&
trigger.parameters.length === 3
) {
trigger.parameters[0].value =
referenceUrl + "webhook_" + trigger.id;
trigger.parameters[1].value = "webhook_" + trigger.id;
// FIXME: Add auth here?
} else {
toast("Something is wrong with the webhook in the copy");
}
}
for (var branchkey in data.branches) {
const branch = data.branches[branchkey];
if (branch.source_id === trigger.id) {
branch.source_id = newId;
}
if (branch.destination_id === trigger.id) {
branch.destination_id = newId;
}
}
trigger.environment = isCloud ? "cloud" : "Shuffle";
trigger.id = newId;
}
}
if (data.actions !== null && data.actions !== undefined) {
for (key in data.actions) {
data.actions[key].authentication_id = "";
for (var subkey in data.actions[key].parameters) {
const param = data.actions[key].parameters[subkey];
if (
param.name.includes("key") ||
param.name.includes("user") ||
param.name.includes("pass") ||
param.name.includes("api") ||
param.name.includes("auth") ||
param.name.includes("secret") ||
param.name.includes("domain") ||
param.name.includes("url") ||
param.name.includes("mail")
) {
// FIXME: This may be a vuln if api-keys are generated that start with $
if (param.value.startsWith("$")) {
console.log("Skipping field, as it's referencing a variable");
} else {
param.value = "";
param.is_valid = false;
}
}
}
const newId = uuidv4();
for (branchkey in data.branches) {
const branch = data.branches[branchkey];
if (branch.source_id === data.actions[key].id) {
branch.source_id = newId;
}
if (branch.destination_id === data.actions[key].id) {
branch.destination_id = newId;
}
}
if (data.actions[key].id === data.start) {
data.start = newId;
}
data.actions[key].environment = "";
data.actions[key].id = newId;
}
}
if (
data.workflow_variables !== null &&
data.workflow_variables !== undefined
) {
for (key in data.workflow_variables) {
const param = data.workflow_variables[key];
if (
param.name.includes("key") ||
param.name.includes("user") ||
param.name.includes("pass") ||
param.name.includes("api") ||
param.name.includes("auth") ||
param.name.includes("secret") ||
param.name.includes("email")
) {
param.value = "";
param.is_valid = false;
}
}
}
return data;
};
const sanitizeWorkflow = (data) => {
data = JSON.parse(JSON.stringify(data));
data["owner"] = "";
console.log("Sanitize start: ", data);
data = deduplicateIds(data);
data["org"] = [];
data["org_id"] = "";
data["execution_org"] = {};
// These are backwards.. True = saved before. Very confuse.
data["previously_saved"] = false;
data["first_save"] = false;
console.log("Sanitize end: ", data);
return data;
};
const exportWorkflow = (data, sanitize) => {
data = JSON.parse(JSON.stringify(data));
let exportFileDefaultName = data.name + ".json";
if (sanitize === true) {
data = sanitizeWorkflow(data);
if (data.subflows !== null && data.subflows !== undefined) {
toast(
"Not exporting with subflows when sanitizing. Please manually export them."
);
data.subflows = [];
}
// for (var key in data.subflows) {
// if (data.sublof
// }
//}
}
// Add correct ID's for triggers
// Add mag
let dataStr = JSON.stringify(data);
let dataUri =
"data:application/json;charset=utf-8," + encodeURIComponent(dataStr);
let linkElement = document.createElement("a");
linkElement.setAttribute("href", dataUri);
linkElement.setAttribute("download", exportFileDefaultName);
linkElement.click();
};
const publishWorkflow = (data) => {
data = JSON.parse(JSON.stringify(data));
data = sanitizeWorkflow(data);
toast("Sanitizing and publishing " + data.name);
// This ALWAYS talks to Shuffle cloud
fetch(globalUrl + "/api/v1/workflows/" + data.id + "/publish", {
method: "POST",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
body: JSON.stringify(data),
credentials: "include",
})
.then((response) => {
if (response.status !== 200) {
console.log("Status not 200 for workflow publish :O!");
} else {
if (isCloud) {
toast("Successfully published workflow");
} else {
toast(
"Successfully published workflow to https://shuffler.io"
);
}
}
return response.json();
})
.then((responseJson) => {
if (responseJson.reason !== undefined) {
toast("Failed publishing: ", responseJson.reason);
}
getAvailableWorkflows();
})
.catch((error) => {
toast(error.toString());
});
};
const copyWorkflow = (data) => {
data = JSON.parse(JSON.stringify(data));
toast("Copying workflow " + data.name);
data.id = "";
data.name = data.name + "_copy";
data = deduplicateIds(data);
fetch(globalUrl + "/api/v1/workflows", {
method: "POST",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
body: JSON.stringify(data),
credentials: "include",
})
.then((response) => {
if (response.status !== 200) {
console.log("Status not 200 for workflows :O!");
return;
}
return response.json();
})
.then(() => {
setTimeout(() => {
getAvailableWorkflows();
}, 1000);
})
.catch((error) => {
toast(error.toString());
});
};
const deleteWorkflow = (id) => {
fetch(globalUrl + "/api/v1/workflows/" + id, {
method: "DELETE",
headers: {
"Content-Type": "application/json",
Accept: "application/json",
},
credentials: "include",
})
.then((response) => {
if (response.status !== 200) {
console.log("Status not 200 for setting workflows :O!");
toast("Failed deleting workflow. Do you have access?");
} else {
toast("Deleted workflow " + id);
}
return response.json();
})
.then(() => {
setTimeout(() => {
getAvailableWorkflows();
}, 1000);
})
.catch((error) => {
toast(error.toString());
});
};
const handleChipClick = (e) => {
addFilter(e.target.innerHTML);
};
const NewWorkflowPaper = () => {
const [hover, setHover] = React.useState(false);
const innerColor = "rgba(255,255,255,0.3)";
const setupPaperStyle = {
minHeight: paperAppStyle.minHeight,
width: paperAppStyle.width,
color: innerColor,
padding: paperAppStyle.padding,
borderRadius: paperAppStyle.borderRadius,
display: "flex",
boxSizing: "border-box",
position: "relative",
border: `2px solid ${innerColor}`,
cursor: "pointer",
backgroundColor: hover ? "rgba(39,41,45,0.5)" : "rgba(39,41,45,1)",
};
return (