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 ? ( { setExportModalOpen(false); setSelectedWorkflow({}); }} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: 500, padding: 30, }, }} >
Want to auto-sanitize this workflow before exporting?
This will make potentially sensitive fields such as username, password, url etc. empty
) : null; const publishModal = publishModalOpen ? ( { setPublishModalOpen(false); setSelectedWorkflow({}); }} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: 500, padding: 50, }, }} >
Are you sure you want to PUBLISH this workflow?
Before publishing, we will sanitize all inputs, remove references to you, randomize ID's and remove your authentication.
) : null; const deleteModal = deleteModalOpen ? ( { setDeleteModalOpen(false); setSelectedWorkflowId(""); }} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: 500, }, }} >
Are you sure you want to delete this workflow?
Other workflows relying on this one may stop working
) : 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 ( setModalOpen(true)} onMouseOver={() => { setHover(true); }} onMouseOut={() => { setHover(false); }} > ); }; const WorkflowPaper = (props) => { const { data } = props; const [open, setOpen] = React.useState(false); const [anchorEl, setAnchorEl] = React.useState(null); var boxColor = "#FECC00"; if (data.is_valid) { boxColor = "#86c142"; } if (!data.previously_saved) { boxColor = "#f85a3e"; } const menuClick = (event) => { setOpen(!open); setAnchorEl(event.currentTarget); }; var parsedName = data.name; if ( parsedName !== undefined && parsedName !== null && parsedName.length > 20 ) { parsedName = parsedName.slice(0, 21) + ".."; } const actions = data.actions !== null ? data.actions.length : 0; const [triggers, subflows] = getWorkflowMeta(data); const workflowMenuButtons = ( { setOpen(false); setAnchorEl(null); }} > { setModalOpen(true); setEditingWorkflow(JSON.parse(JSON.stringify(data))); setNewWorkflowName(data.name); setNewWorkflowDescription(data.description); setDefaultReturnValue(data.default_return_value); if (data.tags !== undefined && data.tags !== null) { setNewWorkflowTags(JSON.parse(JSON.stringify(data.tags))); } }} key={"change"} > {"Change details"} { setSelectedWorkflow(data); setPublishModalOpen(true); }} key={"publish"} > {"Publish Workflow"} { copyWorkflow(data); setOpen(false); }} key={"duplicate"} > {"Duplicate Workflow"} { setExportModalOpen(true); if (data.triggers !== null && data.triggers !== undefined) { var newSubflows = []; for (var key in data.triggers) { const trigger = data.triggers[key]; if ( trigger.parameters !== null && trigger.parameters !== undefined ) { for (var subkey in trigger.parameters) { const param = trigger.parameters[subkey]; if ( param.name === "workflow" && param.value !== data.id && !newSubflows.includes(param.value) ) { newSubflows.push(param.value); } } } } var parsedworkflows = []; for (var key in newSubflows) { const foundWorkflow = workflows.find( (workflow) => workflow.id === newSubflows[key] ); if (foundWorkflow !== undefined && foundWorkflow !== null) { parsedworkflows.push(foundWorkflow); } } if (parsedworkflows.length > 0) { console.log( "Appending subflows during export: ", parsedworkflows.length ); data.subflows = parsedworkflows; } } setExportData(data); setOpen(false); }} key={"export"} > {"Export Workflow"} { setDeleteModalOpen(true); setSelectedWorkflowId(data.id); setOpen(false); }} key={"delete"} > {"Delete Workflow"} ); var image = ""; var orgName = ""; var orgId = ""; if (userdata.orgs !== undefined) { const foundOrg = userdata.orgs.find((org) => org.id === data["org_id"]); if (foundOrg !== undefined && foundOrg !== null) { //position: "absolute", bottom: 5, right: -5, const imageStyle = { width: imagesize, height: imagesize, pointerEvents: "none", marginLeft: data.creator_org !== undefined && data.creator_org.length > 0 ? 20 : 0, borderRadius: 10, border: foundOrg.id === userdata.active_org.id ? `3px solid ${boxColor}` : null, cursor: "pointer", marginRight: 10, }; image = foundOrg.image === "" ? ( {foundOrg.name} ) : ( {foundOrg.name} {}} /> ); orgName = foundOrg.name; orgId = foundOrg.id; } } return (
{ addFilter(orgId); }} > {image}
{parsedName}
{actions} {triggers} { if (subflows === 0) { toast("No subflows for " + data.name); return; } var newWorkflows = [data]; for (var key in data.triggers) { const trigger = data.triggers[key]; if (trigger.app_name !== "Shuffle Workflow") { continue; } if ( trigger.parameters !== undefined && trigger.parameters !== null && trigger.parameters.length > 0 && trigger.parameters[0].name === "workflow" ) { const newWorkflow = workflows.find( (item) => item.id === trigger.parameters[0].value ); if (newWorkflow !== null && newWorkflow !== undefined) { newWorkflows.push(newWorkflow); continue; } } } setFilters(["Subflows of " + data.name]); setFilteredWorkflows(newWorkflows); }} > {subflows} {data.tags !== undefined ? data.tags.map((tag, index) => { if (index >= 3) { return null; } return ( ); }) : null} {data.actions !== undefined && data.actions !== null ? (
{workflowMenuButtons}
) : null}
) } // Can create and set workflows const setNewWorkflow = ( name, description, tags, defaultReturnValue, editingWorkflow, redirect ) => { var method = "POST"; var extraData = ""; var workflowdata = {}; if (editingWorkflow.id !== undefined) { console.log("Building original workflow"); method = "PUT"; extraData = "/" + editingWorkflow.id + "?skip_save=true"; workflowdata = editingWorkflow; console.log("REMOVING OWNER"); workflowdata["owner"] = ""; // FIXME: Loop triggers and turn them off? } workflowdata["name"] = name; workflowdata["description"] = description; if (tags !== undefined) { workflowdata["tags"] = tags; } if (defaultReturnValue !== undefined) { workflowdata["default_return_value"] = defaultReturnValue; } return fetch(globalUrl + "/api/v1/workflows" + extraData, { method: method, headers: { "Content-Type": "application/json", Accept: "application/json", }, body: JSON.stringify(workflowdata), credentials: "include", }) .then((response) => { if (response.status !== 200) { console.log("Status not 200 for workflows :O!"); return; } setSubmitLoading(false); return response.json(); }) .then((responseJson) => { if (responseJson.success === false) { if (responseJson.reason !== undefined) { toast("Error setting workflow: ", responseJson.reason) } else { toast("Error setting workflow.") } return } if (method === "POST" && redirect) { window.location.pathname = "/workflows/" + responseJson["id"]; setModalOpen(false); } else if (!redirect) { // Update :) setTimeout(() => { getAvailableWorkflows(); }, 1000); setImportLoading(false); setModalOpen(false); } else { toast("Successfully changed basic info for workflow"); setModalOpen(false); } return responseJson; }) .catch((error) => { toast(error.toString()); setImportLoading(false); setModalOpen(false); setSubmitLoading(false); }); }; const importFiles = (event) => { console.log("Importing!"); setImportLoading(true); if (event.target.files.length > 0) { for (var key in event.target.files) { const file = event.target.files[key]; if (file.type !== "application/json") { if (file.type !== undefined) { toast("File has to contain valid json"); setImportLoading(false); } continue; } const reader = new FileReader(); // Waits for the read reader.addEventListener("load", (event) => { var data = reader.result; try { data = JSON.parse(reader.result); } catch (e) { toast("Invalid JSON: " + e); setImportLoading(false); 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; data.first_save = false; data.previously_saved = false; data.is_valid = false; // 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()); }); }); // Actually reads reader.readAsText(file); } } setLoadWorkflowsModalOpen(false); }; const getWorkflowMeta = (data) => { let triggers = 0; let subflows = 0; if ( data.triggers !== undefined && data.triggers !== null && data.triggers.length > 0 ) { triggers = data.triggers.length; for (let key in data.triggers) { if (data.triggers[key].app_name === "Shuffle Workflow") { subflows += 1; } } } return [triggers, subflows]; }; const WorkflowListView = () => { let workflowData = ""; if (workflows.length > 0) { const columns = [ { field: "image", headerName: "Logo", width: 50, sortable: false, renderCell: (params) => { const data = params.row.record; var boxColor = "#FECC00"; if (data.is_valid) { boxColor = "#86c142"; } if (!data.previously_saved) { boxColor = "#f85a3e"; } var image = ""; if (userdata.orgs !== undefined) { const foundOrg = userdata.orgs.find( (org) => org.id === data["org_id"] ); if (foundOrg !== undefined && foundOrg !== null) { //position: "absolute", bottom: 5, right: -5, const imageStyle = { width: imagesize + 7, height: imagesize + 7, pointerEvents: "none", marginLeft: data.creator_org !== undefined && data.creator_org.length > 0 ? 20 : 0, borderRadius: 10, border: foundOrg.id === userdata.active_org.id ? `3px solid ${boxColor}` : null, cursor: "pointer", marginTop: 5, }; // image = foundOrg.image === "" ? ( {foundOrg.name} ) : ( {foundOrg.name} { //setFilteredWorkflows(newWorkflows) }} /> ); } } return
{image}
; }, }, { field: "title", headerName: "Title", width: 330, renderCell: (params) => { const data = params.row.record; return ( {data.name} ); }, }, { field: "options", headerName: "Options", width: 200, sortable: false, disableClickEventBubbling: true, renderCell: (params) => { const data = params.row.record; const actions = data.actions !== null ? data.actions.length : 0; let [triggers, subflows] = getWorkflowMeta(data); return (
{actions} {triggers} { if (subflows === 0) { toast("No subflows for " + data.name); return; } var newWorkflows = [data]; for (var key in data.triggers) { const trigger = data.triggers[key]; if (trigger.app_name !== "Shuffle Workflow") { continue; } if ( trigger.parameters !== undefined && trigger.parameters !== null && trigger.parameters.length > 0 && trigger.parameters[0].name === "workflow" ) { const newWorkflow = workflows.find( (item) => item.id === trigger.parameters[0].value ); if ( newWorkflow !== null && newWorkflow !== undefined ) { newWorkflows.push(newWorkflow); continue; } } } setFilters(["Subflows of " + data.name]); setFilteredWorkflows(newWorkflows); }} > {subflows}
); }, }, { field: "tags", headerName: "Tags", maxHeight: 15, width: 300, sortable: false, disableClickEventBubbling: true, renderCell: (params) => { const data = params.row.record; return ( {data.tags !== undefined ? data.tags.map((tag, index) => { if (index >= 3) { return null; } return ( ); }) : null} ); }, }, { field: "", headerName: "", maxHeight: 15, width: 100, sortable: false, disableClickEventBubbling: true, renderCell: (params) => {}, }, ]; let rows = []; rows = filteredWorkflows.map((data, index) => { let obj = { id: index + 1, title: data.name, record: data, }; return obj; }) workflowData = ( ); } return
{workflowData}
; }; const modalView = modalOpen ? ( { setModalOpen(false); }} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: "800px", }, }} >
{editingWorkflow.id !== undefined ? "Editing" : "New"} workflow
setNewWorkflowName(event.target.value)} InputProps={{ style: { color: "white", }, }} color="primary" placeholder="Name" margin="dense" defaultValue={newWorkflowName} autoFocus fullWidth /> setNewWorkflowDescription(event.target.value)} InputProps={{ style: { color: "white", }, }} color="primary" defaultValue={newWorkflowDescription} placeholder="Description" rows="3" multiline margin="dense" fullWidth /> { newWorkflowTags.push(chip); setNewWorkflowTags(newWorkflowTags); }} onDelete={(chip, index) => { newWorkflowTags.splice(index, 1); setNewWorkflowTags(newWorkflowTags); }} /> setDefaultReturnValue(event.target.value)} InputProps={{ style: { color: "white", }, }} color="primary" defaultValue={defaultReturnValue} placeholder="Default return value (used for Subflows if the subflow fails)" rows="3" multiline margin="dense" fullWidth />
) : null; const viewSize = { workflowView: 4, executionsView: 3, executionResults: 4, }; if (viewSize.workflowView === 0) { workflowViewStyle.display = "none"; } const workflowButtons = ( {view === "list" && ( )} {view === "grid" && ( )} {importLoading ? ( ) : ( )} (upload = ref)} onChange={importFiles} /> {workflows.length > 0 ? ( ) : null} {isCloud ? null : ( )} ); const workflowViewStyle = { flex: viewSize.workflowView, margin: "auto", marginTop: 25, textAlign: "center", maxWidth: 600, }; const WorkflowView = () => { var workflowDelay = -150 var appDelay = -75 const textSpacingDiff = 8 const textType = "body2" // Discover use-cases made by us and other creators! const steps = [ { html: ( { if (isCloud) { ReactGA.event({ category: "getting-started", action: `integerations_find_click`, }) } }}> Find relevant apps and start your automation journey ), tutorial: "find_integrations", }, { html: Discover Use-Case ideas and  { if (isCloud) { navigate(`/search?tab=workflows`) ReactGA.event({ category: "getting-started", action: `workflow_find_click`, }) return } else { toast("TBD: Coming in version 1.0.0"); } const ele = document.getElementById("shuffle_search_field") if (ele !== undefined && ele !== null) { console.log("Found ele: ", ele) ele.focus() ele.style.borderColor = "#f86a3e" ele.style.borderWidth = "2px" } else { //toast("TBD: Coming in version 1.0.0"); } }}> workflows made by other creators! , tutorial: "discover_workflows", }, { html: ( { if (isCloud) { ReactGA.event({ category: "getting-started", action: `create_workflow_click`, }) } }}> Learn to use Shuffle by  {setModalOpen(true)}}> creating your first workflow and reading the docs. ), tutorial: "learn_shuffle", }, { html: Configure your organization name in the admin panel and invite your team , tutorial: "configure_organization", } ] return (
{ setVideoViewOpen(false) }} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: 560, minHeight: 415, textAlign: "center", }, }} > Welcome to Shuffle! { e.preventDefault(); setVideoViewOpen(false) }} >
Getting Started with Shuffle We provide everything you need to automate your operations - apps, default workflows, security dashboards and analytics that work well together. {steps.map((data, index) => { var tutorialFound = false if (userdata.tutorials !== undefined && userdata.tutorials !== null && userdata.tutorials.length > 0 && data.tutorial !== undefined) { const foundTutorial = userdata.tutorials.find(tutorial => tutorial === data.tutorial) if (foundTutorial !== undefined && foundTutorial !== null) { console.log("Found tutorial for ", data.tutorial) tutorialFound = true } if (tutorialFound === false) { if (data.tutorial === "discover_workflows") { if (workflows.length > 0) { for (var key in workflows) { const tmpworkflow = workflows[key] if (tmpworkflow.published_id !== undefined && tmpworkflow.published_id !== null && tmpworkflow.published_id.length > 0) { tutorialFound = true break } } } } if (data.tutorial === "learn_shuffle") { //tutorial: "discover_workflows", if (workflows.length > 0) { tutorialFound = true } } if (data.tutorial === "configure_organization") { if (userdata.active_org.name !== userdata.username) { tutorialFound = true } } } } return (
{tutorialFound ? : {index+1} }
{data.html}
) })}
Invite your team Get teammates, managers and customers involved.
Need help? We help with automation, scaling, training and more. Get involved!
{/*
Need assistance? Ask our support team (it's free!).
*/}
{/*
{workflows.length}
ACTIVE WORKFLOWS
{workflows.length}
AVAILABE WORKFLOWS
{workflows.length}
NOTIFICATIONS
*/} {/* chipRenderer={({ value, isFocused, isDisabled, handleClick, handleRequestDelete }, key) => { console.log("VALUE: ", value) return ( {value} ) }} */} {/*
{ addFilter(chip); }} onDelete={(_, index) => { removeFilter(index); }} />
{workflowButtons}
{actionImageList !== undefined && actionImageList !== null && actionImageList.length > 0 ? (
{actionImageList.map((data, index) => { if ( data.large_image === undefined || data.large_image === null || data.large_image.length === 0 ) { return null; } if (data.app_name.toLowerCase() === "shuffle tools") { data.large_image = theme.palette.defaultImage; } if (firstLoad) { appDelay += 75 } else { appDelay = 0 } return ( { console.log("FILTER: ", data); addFilter(data.app_name); }} >
{data.app_name}
); })}
) : null} {view === "grid" ? ( {filteredWorkflows.map((data, index) => { if (firstLoad) { workflowDelay += 75 } else { workflowDelay = 0 } return ( ) })} ) : ( )}
*/}
); }; const importWorkflowsFromUrl = (url) => { console.log("IMPORT WORKFLOWS FROM ", downloadUrl); const parsedData = { url: url, field_3: downloadBranch || "master", }; if (field1.length > 0) { parsedData["field_1"] = field1; } if (field2.length > 0) { parsedData["field_2"] = field2; } toast("Getting specific workflows from your URL."); fetch(globalUrl + "/api/v1/workflows/download_remote", { method: "POST", mode: "cors", headers: { Accept: "application/json", }, body: JSON.stringify(parsedData), credentials: "include", }) .then((response) => { if (response.status === 200) { toast("Successfully loaded workflows from " + downloadUrl); setTimeout(() => { getAvailableWorkflows(); }, 1000); } return response.json(); }) .then((responseJson) => { if (!responseJson.success) { if (responseJson.reason !== undefined) { toast("Failed loading: " + responseJson.reason); } else { toast("Failed loading"); } } }) .catch((error) => { toast(error.toString()); }); }; const handleGithubValidation = () => { importWorkflowsFromUrl(downloadUrl); setLoadWorkflowsModalOpen(false); }; const workflowDownloadModalOpen = loadWorkflowsModalOpen ? ( {}} PaperProps={{ style: { backgroundColor: surfaceColor, color: "white", minWidth: "800px", minHeight: "320px", }, }} >
Load workflows from github repo
Repository (supported: github, gitlab, bitbucket) setDownloadUrl(e.target.value)} placeholder="https://github.com/frikky/shuffle-apps" fullWidth /> Branch (default value is "master"):
setDownloadBranch(e.target.value)} placeholder="master" fullWidth />
Authentication (optional - private repos etc):
setField1(e.target.value)} type="username" placeholder="Username / APIkey (optional)" fullWidth /> setField2(e.target.value)} type="password" placeholder="Password (optional)" fullWidth />
) : null; const loadedCheck = isLoaded && isLoggedIn && workflowDone ? (
{/* */} 1366 ? 1366 : 1200, margin: "auto", padding: 20, }} onDrop={uploadFile} > {modalView} {deleteModal} {exportVerifyModal} {publishModal} {workflowDownloadModalOpen}
) : (
Loading Workflows
); // Maybe use gridview or something, idk return
{loadedCheck}
; }; export default GettingStarted;