mirror of
				https://github.com/bunny-lab-io/Borealis.git
				synced 2025-10-26 15:21:57 -06:00 
			
		
		
		
	Merge branch 'codex/adjust-layout-and-spacing-of-variables-section' of https://github.com/bunny-lab-io/Borealis into codex/adjust-layout-and-spacing-of-variables-section
This commit is contained in:
		| @@ -5,14 +5,6 @@ | ||||
|   "category": "script", | ||||
|   "type": "powershell", | ||||
|   "script": "# Define the file path\n$filePath = \"C:\\Canary.txt\"\n\n# Write some content into the file\n\"SYSTEM Canary is alive.\"ddss | Out-File -FilePath $filePath -Encoding UTF8\n", | ||||
|   "script_lines": [ | ||||
|     "# Define the file path", | ||||
|     "$filePath = \"C:\\Canary.txt\"", | ||||
|     "", | ||||
|     "# Write some content into the file", | ||||
|     "\"SYSTEM Canary is alive.\"ddss | Out-File -FilePath $filePath -Encoding UTF8", | ||||
|     "" | ||||
|   ], | ||||
|   "timeout_seconds": 3600, | ||||
|   "sites": { | ||||
|     "mode": "specific", | ||||
|   | ||||
| @@ -212,13 +212,11 @@ function fromServerDocument(doc = {}, defaultType = "powershell") { | ||||
|     assembly.description = doc.description || ""; | ||||
|     assembly.category = doc.category || assembly.category; | ||||
|     assembly.type = doc.type || assembly.type; | ||||
|     if (Array.isArray(doc.script_lines)) { | ||||
|       assembly.script = doc.script_lines | ||||
|         .map((line) => (line == null ? "" : String(line))) | ||||
|         .join("\n"); | ||||
|     } else { | ||||
|       assembly.script = doc.script ?? doc.content ?? ""; | ||||
|     } | ||||
|     const legacyScript = Array.isArray(doc.script_lines) | ||||
|       ? doc.script_lines.map((line) => (line == null ? "" : String(line))).join("\n") | ||||
|       : ""; | ||||
|     const script = doc.script ?? doc.content ?? legacyScript; | ||||
|     assembly.script = typeof script === "string" ? script : legacyScript; | ||||
|     const timeout = doc.timeout_seconds ?? doc.timeout ?? assembly.timeoutSeconds; | ||||
|     assembly.timeoutSeconds = Number.isFinite(Number(timeout)) | ||||
|       ? Number(timeout) | ||||
| @@ -238,7 +236,6 @@ function toServerDocument(assembly) { | ||||
|   const normalizedScript = typeof assembly.script === "string" | ||||
|     ? assembly.script.replace(/\r\n/g, "\n") | ||||
|     : ""; | ||||
|   const scriptLines = normalizedScript ? normalizedScript.split("\n") : []; | ||||
|   const timeoutNumeric = Number(assembly.timeoutSeconds); | ||||
|   const timeoutSeconds = Number.isFinite(timeoutNumeric) ? Math.max(0, Math.round(timeoutNumeric)) : 3600; | ||||
|   return { | ||||
| @@ -248,7 +245,6 @@ function toServerDocument(assembly) { | ||||
|     category: assembly.category || "script", | ||||
|     type: assembly.type || "powershell", | ||||
|     script: normalizedScript, | ||||
|     script_lines: scriptLines, | ||||
|     timeout_seconds: timeoutSeconds, | ||||
|     sites: { | ||||
|       mode: assembly.sites?.mode === "specific" ? "specific" : "all", | ||||
| @@ -728,12 +724,16 @@ export default function AssemblyEditor({ | ||||
|                 value={assembly.description} | ||||
|                 onChange={(e) => updateAssembly({ description: e.target.value })} | ||||
|                 multiline | ||||
|                 minRows={3} | ||||
|                 minRows={2} | ||||
|                 maxRows={8} | ||||
|                 fullWidth | ||||
|                 variant="outlined" | ||||
|                 sx={{ | ||||
|                   ...INPUT_BASE_SX, | ||||
|                   "& .MuiOutlinedInput-inputMultiline": { padding: "4px 8px" } | ||||
|                   "& .MuiOutlinedInput-inputMultiline": { | ||||
|                     padding: "6px 12px", | ||||
|                     lineHeight: 1.4 | ||||
|                   } | ||||
|                 }} | ||||
|               /> | ||||
|             </Grid> | ||||
| @@ -906,109 +906,162 @@ export default function AssemblyEditor({ | ||||
|                       sx={{ | ||||
|                         display: "flex", | ||||
|                         flexWrap: "wrap", | ||||
|                         alignItems: "center", | ||||
|                         gap: 2 | ||||
|                         alignItems: "stretch", | ||||
|                         gap: { xs: 2, lg: 1.5 }, | ||||
|                         pt: 1 | ||||
|                       }} | ||||
|                     > | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", md: "0 1 23%" } }}> | ||||
|                         <TextField | ||||
|                           label="Variable Name" | ||||
|                           value={variable.name} | ||||
|                           onChange={(e) => updateVariable(variable.id, { name: e.target.value })} | ||||
|                           fullWidth | ||||
|                           variant="outlined" | ||||
|                           sx={INPUT_BASE_SX} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", md: "0 1 18%" } }}> | ||||
|                         <TextField | ||||
|                           label="Display Label" | ||||
|                           value={variable.label} | ||||
|                           onChange={(e) => updateVariable(variable.id, { label: e.target.value })} | ||||
|                           fullWidth | ||||
|                           variant="outlined" | ||||
|                           sx={INPUT_BASE_SX} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", md: "0 1 18%" } }}> | ||||
|                         <TextField | ||||
|                           select | ||||
|                           fullWidth | ||||
|                           label="Type" | ||||
|                           value={variable.type} | ||||
|                           onChange={(e) => updateVariable(variable.id, { type: e.target.value })} | ||||
|                           sx={SELECT_BASE_SX} | ||||
|                           SelectProps={{ MenuProps: MENU_PROPS }} | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", lg: "0 1 30%" }, minWidth: { lg: 220 } }}> | ||||
|                         <Tooltip | ||||
|                           title="This is the name of the variable you will be referencing inside of the script.  Within the script you will reference this variable with a prefixed \"$env:<variable>\".  For example, a variable named \"message\" would be written in the script as \"$env:message\"." | ||||
|                           arrow | ||||
|                           placement="top-start" | ||||
|                         > | ||||
|                           {VARIABLE_TYPE_OPTIONS.map((opt) => ( | ||||
|                             <MenuItem key={opt.key} value={opt.key}>{opt.label}</MenuItem> | ||||
|                           ))} | ||||
|                         </TextField> | ||||
|                           <TextField | ||||
|                             label="Variable" | ||||
|                             value={variable.name} | ||||
|                             onChange={(e) => updateVariable(variable.id, { name: e.target.value })} | ||||
|                             fullWidth | ||||
|                             variant="outlined" | ||||
|                             sx={INPUT_BASE_SX} | ||||
|                           /> | ||||
|                         </Tooltip> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", lg: "0 1 22%" }, minWidth: { lg: 180 } }}> | ||||
|                         <Tooltip | ||||
|                           title="This is the name that will be given to the variable and seen by Borealis server operators." | ||||
|                           arrow | ||||
|                           placement="top-start" | ||||
|                         > | ||||
|                           <TextField | ||||
|                             label="Display Label" | ||||
|                             value={variable.label} | ||||
|                             onChange={(e) => updateVariable(variable.id, { label: e.target.value })} | ||||
|                             fullWidth | ||||
|                             variant="outlined" | ||||
|                             sx={INPUT_BASE_SX} | ||||
|                           /> | ||||
|                         </Tooltip> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", lg: "0 1 22%" }, minWidth: { lg: 160 } }}> | ||||
|                         <Tooltip | ||||
|                           title="This defines the type of variable data the script should expect." | ||||
|                           arrow | ||||
|                           placement="top-start" | ||||
|                         > | ||||
|                           <TextField | ||||
|                             select | ||||
|                             fullWidth | ||||
|                             label="Type" | ||||
|                             value={variable.type} | ||||
|                             onChange={(e) => updateVariable(variable.id, { type: e.target.value })} | ||||
|                             sx={SELECT_BASE_SX} | ||||
|                             SelectProps={{ MenuProps: MENU_PROPS }} | ||||
|                           > | ||||
|                             {VARIABLE_TYPE_OPTIONS.map((opt) => ( | ||||
|                               <MenuItem key={opt.key} value={opt.key}>{opt.label}</MenuItem> | ||||
|                             ))} | ||||
|                           </TextField> | ||||
|                         </Tooltip> | ||||
|                       </Box> | ||||
|                       <Box | ||||
|                         sx={{ | ||||
|                           flex: { xs: "1 1 100%", md: "0 1 27%" }, | ||||
|                           flex: { xs: "1 1 100%", lg: "0 1 40%" }, | ||||
|                           minWidth: { lg: 220 }, | ||||
|                           display: "flex", | ||||
|                           alignItems: "center" | ||||
|                         }} | ||||
|                       > | ||||
|                         {variable.type === "boolean" ? ( | ||||
|                           <FormControlLabel | ||||
|                             control={ | ||||
|                               <Checkbox | ||||
|                                 checked={Boolean(variable.defaultValue)} | ||||
|                                 onChange={(e) => updateVariable(variable.id, { defaultValue: e.target.checked })} | ||||
|                                 sx={{ color: "#58a6ff" }} | ||||
|                               /> | ||||
|                             } | ||||
|                             label="Default Value" | ||||
|                             sx={{ | ||||
|                               color: "#9ba3b4", | ||||
|                               m: 0, | ||||
|                               "& .MuiFormControlLabel-label": { fontSize: "0.95rem" } | ||||
|                             }} | ||||
|                           /> | ||||
|                           <Tooltip | ||||
|                             title="This is the value that will be pre-populated in the assembly when ran.  Use a sensible default value." | ||||
|                             arrow | ||||
|                             placement="top-start" | ||||
|                           > | ||||
|                             <FormControlLabel | ||||
|                               control={ | ||||
|                                 <Checkbox | ||||
|                                   checked={Boolean(variable.defaultValue)} | ||||
|                                   onChange={(e) => updateVariable(variable.id, { defaultValue: e.target.checked })} | ||||
|                                   sx={{ color: "#58a6ff" }} | ||||
|                                 /> | ||||
|                               } | ||||
|                               label="Default Value" | ||||
|                               sx={{ | ||||
|                                 color: "#9ba3b4", | ||||
|                                 m: 0, | ||||
|                                 "& .MuiFormControlLabel-label": { fontSize: "0.95rem" } | ||||
|                               }} | ||||
|                             /> | ||||
|                           </Tooltip> | ||||
|                         ) : ( | ||||
|                           <TextField | ||||
|                             label="Default Value" | ||||
|                             value={variable.defaultValue ?? ""} | ||||
|                             onChange={(e) => updateVariable(variable.id, { defaultValue: e.target.value })} | ||||
|                             fullWidth | ||||
|                             variant="outlined" | ||||
|                             sx={INPUT_BASE_SX} | ||||
|                           /> | ||||
|                           <Tooltip | ||||
|                             title="This is the value that will be pre-populated in the assembly when ran.  Use a sensible default value." | ||||
|                             arrow | ||||
|                             placement="top-start" | ||||
|                           > | ||||
|                             <TextField | ||||
|                               label="Default Value" | ||||
|                               value={variable.defaultValue ?? ""} | ||||
|                               onChange={(e) => updateVariable(variable.id, { defaultValue: e.target.value })} | ||||
|                               fullWidth | ||||
|                               variant="outlined" | ||||
|                               sx={INPUT_BASE_SX} | ||||
|                             /> | ||||
|                           </Tooltip> | ||||
|                         )} | ||||
|                       </Box> | ||||
|                       <Box | ||||
|                         sx={{ | ||||
|                           flex: { xs: "1 1 100%", md: "0 0 6%" }, | ||||
|                           flex: { xs: "1 1 100%", lg: "0 0 100px" }, | ||||
|                           minWidth: { lg: 100 }, | ||||
|                           display: "flex", | ||||
|                           flexDirection: "column", | ||||
|                           alignItems: "center", | ||||
|                           justifyContent: "center", | ||||
|                           alignItems: "center" | ||||
|                           gap: 0.5 | ||||
|                         }} | ||||
|                       > | ||||
|                         <Tooltip title="Required"> | ||||
|                           <Checkbox | ||||
|                             checked={Boolean(variable.required)} | ||||
|                             onChange={(e) => updateVariable(variable.id, { required: e.target.checked })} | ||||
|                             sx={{ color: "#58a6ff" }} | ||||
|                         <Typography variant="caption" sx={{ color: "#9ba3b4", fontSize: "0.75rem" }}> | ||||
|                           Required | ||||
|                         </Typography> | ||||
|                         <Checkbox | ||||
|                           checked={Boolean(variable.required)} | ||||
|                           onChange={(e) => updateVariable(variable.id, { required: e.target.checked })} | ||||
|                           sx={{ color: "#58a6ff", p: 0.5 }} | ||||
|                           inputProps={{ "aria-label": "Required" }} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", lg: "1 1 0%" }, minWidth: { lg: 220 } }}> | ||||
|                         <Tooltip | ||||
|                           title="Instruct the operator in why this variable exists and how to set it appropriately." | ||||
|                           arrow | ||||
|                           placement="top-start" | ||||
|                         > | ||||
|                           <TextField | ||||
|                             label="Description" | ||||
|                             value={variable.description} | ||||
|                             onChange={(e) => updateVariable(variable.id, { description: e.target.value })} | ||||
|                             fullWidth | ||||
|                             variant="outlined" | ||||
|                             sx={INPUT_BASE_SX} | ||||
|                           /> | ||||
|                         </Tooltip> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: { xs: "1 1 100%", md: "1 1 0%" }, minWidth: { md: 120 } }}> | ||||
|                         <TextField | ||||
|                           label="Description" | ||||
|                           value={variable.description} | ||||
|                           onChange={(e) => updateVariable(variable.id, { description: e.target.value })} | ||||
|                           fullWidth | ||||
|                           variant="outlined" | ||||
|                           sx={INPUT_BASE_SX} | ||||
|                         /> | ||||
|                       </Box> | ||||
|                       <Box sx={{ flex: "1 1 100%", display: "flex", justifyContent: "flex-end" }}> | ||||
|                         <IconButton onClick={() => removeVariable(variable.id)} sx={{ color: "#ff6b6b" }}> | ||||
|                           <DeleteIcon /> | ||||
|                         </IconButton> | ||||
|                       <Box | ||||
|                         sx={{ | ||||
|                           flex: { xs: "1 1 100%", lg: "0 0 auto" }, | ||||
|                           display: "flex", | ||||
|                           justifyContent: "flex-end", | ||||
|                           alignItems: "center", | ||||
|                           ml: { lg: "auto" } | ||||
|                         }} | ||||
|                       > | ||||
|                         <Tooltip title="Remove this Variable." arrow> | ||||
|                           <IconButton onClick={() => removeVariable(variable.id)} sx={{ color: "#ff6b6b" }}> | ||||
|                             <DeleteIcon /> | ||||
|                           </IconButton> | ||||
|                         </Tooltip> | ||||
|                       </Box> | ||||
|                     </Box> | ||||
|                   </Paper> | ||||
|   | ||||
| @@ -182,7 +182,6 @@ class JobScheduler: | ||||
|             "category": "application" if default_type == "ansible" else "script", | ||||
|             "type": default_type, | ||||
|             "script": "", | ||||
|             "script_lines": [], | ||||
|             "variables": [], | ||||
|             "files": [], | ||||
|             "timeout_seconds": 3600, | ||||
| @@ -217,7 +216,6 @@ class JobScheduler: | ||||
|                         doc["script"] = content_val | ||||
|                 normalized_script = (doc["script"] or "").replace("\r\n", "\n") | ||||
|                 doc["script"] = normalized_script | ||||
|                 doc["script_lines"] = normalized_script.split("\n") if normalized_script else [] | ||||
|                 try: | ||||
|                     timeout_raw = data.get("timeout_seconds", data.get("timeout")) | ||||
|                     if timeout_raw is None: | ||||
| @@ -271,7 +269,6 @@ class JobScheduler: | ||||
|             content = "" | ||||
|         normalized_script = (content or "").replace("\r\n", "\n") | ||||
|         doc["script"] = normalized_script | ||||
|         doc["script_lines"] = normalized_script.split("\n") if normalized_script else [] | ||||
|         return doc | ||||
|  | ||||
|     def _ansible_root(self) -> str: | ||||
|   | ||||
| @@ -682,7 +682,6 @@ def _empty_assembly_document(default_type: str = "powershell") -> Dict[str, Any] | ||||
|         "category": "application" if (default_type or "").lower() == "ansible" else "script", | ||||
|         "type": default_type or "powershell", | ||||
|         "script": "", | ||||
|         "script_lines": [], | ||||
|         "timeout_seconds": 3600, | ||||
|         "sites": {"mode": "all", "values": []}, | ||||
|         "variables": [], | ||||
| @@ -718,7 +717,6 @@ def _normalize_assembly_document(obj: Any, default_type: str, base_name: str) -> | ||||
|             doc["script"] = content_val | ||||
|     normalized_script = (doc["script"] or "").replace("\r\n", "\n") | ||||
|     doc["script"] = normalized_script | ||||
|     doc["script_lines"] = normalized_script.split("\n") if normalized_script else [] | ||||
|     timeout_val = obj.get("timeout_seconds", obj.get("timeout")) | ||||
|     if timeout_val is not None: | ||||
|         try: | ||||
| @@ -798,7 +796,6 @@ def _load_assembly_document(abs_path: str, island: str, type_hint: str = "") -> | ||||
|     doc["name"] = base_name | ||||
|     normalized_script = (content or "").replace("\r\n", "\n") | ||||
|     doc["script"] = normalized_script | ||||
|     doc["script_lines"] = normalized_script.split("\n") if normalized_script else [] | ||||
|     if default_type == "ansible": | ||||
|         doc["category"] = "application" | ||||
|     return doc | ||||
|   | ||||
		Reference in New Issue
	
	Block a user