diff --git a/public/version_latest.txt b/public/version_latest.txt index 1f1ac7c2f330..28cbf7c0aae3 100644 --- a/public/version_latest.txt +++ b/public/version_latest.txt @@ -1 +1 @@ -4.9.1 \ No newline at end of file +5.0.0 \ No newline at end of file diff --git a/src/_nav.jsx b/src/_nav.jsx index 77ed1af3a328..0f140ce656a0 100644 --- a/src/_nav.jsx +++ b/src/_nav.jsx @@ -239,6 +239,11 @@ const _nav = [ name: 'Deploy CA Policies', to: '/tenant/conditional/deploy', }, + { + component: CNavItem, + name: 'CA Vacation Mode', + to: '/tenant/conditional/deploy-vacation', + }, { component: CNavItem, name: 'CA Templates', diff --git a/src/components/forms/RFFComponents.jsx b/src/components/forms/RFFComponents.jsx index 4da57f18e54c..2d315826b4f2 100644 --- a/src/components/forms/RFFComponents.jsx +++ b/src/components/forms/RFFComponents.jsx @@ -137,10 +137,17 @@ export const RFFCFormInput = ({ disabled = false, spellCheck = true, autoFocus = false, + onChange, }) => { return ( {({ input, meta }) => { + const handleChange = onChange + ? (e) => { + input.onChange(e) + onChange(e) + } + : input.onChange return (
{label && {label}} @@ -155,6 +162,7 @@ export const RFFCFormInput = ({ placeholder={placeholder} spellCheck={spellCheck} autoFocus={autoFocus} + onChange={handleChange} />
diff --git a/src/components/layout/AppHeader.jsx b/src/components/layout/AppHeader.jsx index 3ceb14542dff..f31d73a124e0 100644 --- a/src/components/layout/AppHeader.jsx +++ b/src/components/layout/AppHeader.jsx @@ -16,17 +16,8 @@ import { } from '@coreui/react' import { AppHeaderSearch } from 'src/components/header' import { TenantSelector } from '../utilities' -import cyberdrainlogolight from 'src/assets/images/CIPP.png' -import cyberdrainlogodark from 'src/assets/images/CIPP_Dark.png' - import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' -import { - faBars, - faCaretSquareLeft, - faCaretSquareRight, - faHamburger, - faStroopwafel, -} from '@fortawesome/free-solid-svg-icons' +import { faBars } from '@fortawesome/free-solid-svg-icons' import { setCurrentTheme, setUserSettings, toggleSidebarShow } from 'src/store/features/app' import { useMediaPredicate } from 'react-media-hook' import { useGenericGetRequestQuery, useLoadAlertsDashQuery } from 'src/store/api/app' diff --git a/src/components/layout/AppSidebar.jsx b/src/components/layout/AppSidebar.jsx index aad3a2eb170d..f680f42116d1 100644 --- a/src/components/layout/AppSidebar.jsx +++ b/src/components/layout/AppSidebar.jsx @@ -6,21 +6,22 @@ import { CImage, CSidebar, CSidebarBrand, - CSidebarHeader, CSidebarNav, } from '@coreui/react' import { AppSidebarNav } from 'src/components/layout' import SimpleBar from 'simplebar-react' import 'simplebar/dist/simplebar.min.css' import navigation from 'src/_nav' -import { setSidebarVisible } from 'src/store/features/app' -import cyberdrainlogolight from 'src/assets/images/CIPP.png' const AppSidebar = () => { + const i = + '' + const dispatch = useDispatch() - const unfoldable = useSelector((state) => state.app.sidebarUnfoldable) const sidebarShow = useSelector((state) => state.app.sidebarShow) - + if (!i.includes('JGySCBt1QXmNc')) { + throw '' + } return ( { @@ -31,7 +32,7 @@ const AppSidebar = () => { visible={sidebarShow} > - + +
{String(cell)}
+ + ) +} +export const cellMathFormatter = + ({ col } = {}) => + (row) => { + const evaluateCalculation = (calculation, row) => { + try { + const formattedCalculation = calculation.replace(/\b\w+(\.\w+|\[\d+\])*\b/g, (key) => { + if (!isNaN(key)) { + return parseFloat(key) + } + + const path = key.split(/\.|\[(\d+)\]/).filter(Boolean) // Splits keys and array indices + let currentObject = row + for (const prop of path) { + if (currentObject && prop in currentObject) { + currentObject = currentObject[prop] + } else if (!isNaN(prop)) { + // Checks if the prop is an array index + currentObject = currentObject[parseInt(prop, 10)] + } else { + throw new Error(`Property '${prop}' not found in row`) + } + } + + return parseFloat(currentObject) + }) + + return Number(eval(formattedCalculation)) + } catch (e) { + console.error(e) + return null + } + } + + const result = evaluateCalculation(col.value, row) + + if (result === null) { + return 'N/A' + } + + if (col.showAs === 'percentage') { + return `${result.toFixed(2)}%` + } else { + return result.toFixed(2) + } + } + +export default cellMathFormatter diff --git a/src/components/utilities/CippActionsOffcanvas.jsx b/src/components/utilities/CippActionsOffcanvas.jsx index 20e84a76b0bc..3cec2692a969 100644 --- a/src/components/utilities/CippActionsOffcanvas.jsx +++ b/src/components/utilities/CippActionsOffcanvas.jsx @@ -38,6 +38,34 @@ export default function CippActionsOffcanvas(props) { } const handleModal = useCallback( (modalMessage, modalUrl, modalType = 'GET', modalBody, modalInput, modalDropdown) => { + const handlePostConfirm = () => { + const selectedValue = inputRef.current.value + console.log(inputRef) + let additionalFields = {} + + if (inputRef.current.nodeName === 'SELECT') { + const selectedItem = dropDownInfo.data.find( + (item) => item[modalDropdown.valueField] === selectedValue, + ) + if (selectedItem && modalDropdown.addedField) { + Object.keys(modalDropdown.addedField).forEach((key) => { + additionalFields[key] = selectedItem[modalDropdown.addedField[key]] + }) + } + } + const postRequestBody = { + ...modalBody, + ...additionalFields, + input: selectedValue, + } + // Send the POST request + genericPostRequest({ + path: modalUrl, + values: postRequestBody, + }) + } + + // Modal setup for GET, codeblock, and other types if (modalType === 'GET') { ModalService.confirm({ body: ( @@ -82,12 +110,7 @@ export default function CippActionsOffcanvas(props) { ), title: 'Confirm', - onConfirm: () => [ - genericPostRequest({ - path: modalUrl, - values: { ...modalBody, ...{ input: inputRef.current.value } }, - }), - ], + onConfirm: handlePostConfirm, }) } }, @@ -99,7 +122,6 @@ export default function CippActionsOffcanvas(props) { modalContent, ], ) - useEffect(() => { if (dropDownInfo.isFetching) { handleModal( diff --git a/src/data/BPAField.schema.v1.json b/src/data/BPAField.schema.v1.json index 56a6a07fd1de..810329db4e16 100644 --- a/src/data/BPAField.schema.v1.json +++ b/src/data/BPAField.schema.v1.json @@ -87,6 +87,10 @@ { "const": "number", "title": "Displays as a numerical value" + }, + { + "const": "math", + "title": "Displays as a calculated value" } ] } diff --git a/src/data/standards.json b/src/data/standards.json index 51301804bb01..4781d1075fa4 100644 --- a/src/data/standards.json +++ b/src/data/standards.json @@ -37,12 +37,30 @@ { "name": "standards.AuditLog", "cat": "Global Standards", - "helpText": "Enables the Unified Audit Log for tracking and auditing activities; also runs Enable-OrganizationCustomization if necessary.", + "helpText": "Enables the Unified Audit Log for tracking and auditing activities. Also runs Enable-OrganizationCustomization if necessary.", "addedComponent": [], "label": "Enable the Unified Audit Log", "impact": "Low Impact", "impactColour": "info" }, + { + "name": "standards.PhishProtection", + "cat": "Global Standards", + "helpText": "Adds branding to the logon page that only appears if the url is not login.microsoftonline.com. This potentially prevents AITM attacks via EvilNginx. This will also automatically generate alerts if a clone of your login page has been found. (P1 or higher required)", + "addedComponent": [], + "label": "Enable Phishing Protection system via branding CSS", + "impact": "Low Impact", + "impactColour": "info" + }, + { + "name": "standards.EnableCustomerLockbox", + "cat": "Global Standards", + "helpText": "Enables Customer Lockbox that offers an approval process for Microsoft support to access organization data", + "addedComponent": [], + "label": "Enable Customer Lockbox", + "impact": "Low Impact", + "impactColour": "info" + }, { "name": "standards.AnonReportDisable", "cat": "Global Standards", @@ -55,7 +73,7 @@ { "name": "standards.DisableGuestDirectory", "cat": "Global Standards", - "helpText": "Disables Guest access to enumerate directory objects. This prevents guest users from see other users or guests in the directory.", + "helpText": "Disables Guest access to enumerate directory objects. This prevents guest users from seeing other users or guests in the directory.", "addedComponent": [], "label": "Restrict guest user access to directory objects", "impact": "Low Impact", @@ -64,7 +82,7 @@ { "name": "standards.DisableBasicAuthSMTP", "cat": "Global Standards", - "helpText": "Disables SMTP AUTH for the organization. This is the default for new tenants. Sets the entire tenant to no longer allow SMTP AUTH, and as such has no exclusions.", + "helpText": "Disables SMTP AUTH for the organization and all users. This is the default for new tenants. ", "addedComponent": [], "label": "Disable SMTP Basic Authentication", "impact": "Medium Impact", @@ -84,7 +102,7 @@ "cat": "Entra (AAD) Standards", "helpText": "Enables the tenant to use LAPS. You must still create a policy for LAPS to be active on all devices. Use the template standards to deploy this by default.", "addedComponent": [], - "label": "Enable LAPs on the tenant", + "label": "Enable LAPS on the tenant", "impact": "Low Impact", "impactColour": "info" }, @@ -102,7 +120,7 @@ "name": "standards.allowOTPTokens", "helpText": "Allows you to use MS authenticator OTP token generator", "addedComponent": [], - "label": "Enable OTP via Authenticator.", + "label": "Enable OTP via Authenticator", "impact": "Low Impact", "impactColour": "info" }, @@ -221,7 +239,7 @@ "name": "standards.NudgeMFA.enable", "helpText": "Enables registration campaign for the tenant", "addedComponent": [], - "label": "Request to setup Authenticator if not setup yet.", + "label": "Request to setup Authenticator if not setup yet", "impact": "Low Impact", "impactColour": "info" }, @@ -230,7 +248,7 @@ "name": "standards.NudgeMFA.disable", "helpText": "Disables registration campaign for the tenant", "addedComponent": [], - "label": "Disables the request to setup Authenticator if setup.", + "label": "Disables the request to setup Authenticator if setup", "impact": "Low Impact", "impactColour": "info" }, @@ -290,16 +308,16 @@ "label": "Allowed application IDs, comma separated" } ], - "label": "Require admin consent for applications (Prevent OAuth phishing.)", - "impact": "Medium impact", + "label": "Require admin consent for applications (Prevent OAuth phishing)", + "impact": "Medium Impact", "impactColour": "warning" }, { "cat": "Entra (AAD) Standards", "name": "standards.OauthConsentLowSec", "helpText": "Sets the default oauth consent level so users can consent to applications that have low risks.", - "label": "Allow users to consent to applications with low security risk (Prevent OAuth phishing. Lower impact, less secure.)", - "impact": "Medium impact", + "label": "Allow users to consent to applications with low security risk (Prevent OAuth phishing. Lower impact, less secure)", + "impact": "Medium Impact", "impactColour": "warning" }, { @@ -364,7 +382,7 @@ { "name": "standards.OutBoundSpamAlert", "cat": "Exchange Standards", - "helpText": "Set the Outbound Spam Alert e-mail address.", + "helpText": "Set the Outbound Spam Alert e-mail address", "addedComponent": [ { "type": "input", @@ -406,7 +424,23 @@ ] } ], - "label": "Enable or disable 'external' warning in Outlook.", + "label": "Enable or disable 'external' warning in Outlook", + "impact": "Low Impact", + "impactColour": "info" + }, + { + "name": "standards.EnableMailTips", + "cat": "Exchange Standards", + "helpText": "Enables all MailTips in Outlook. MailTips are the notifications Outlook and Outlook on the web shows when an email you create, meets some requirements", + "addedComponent": [ + { + "type": "number", + "name": "standards.EnableMailTips.MailTipsLargeAudienceThreshold", + "label": "Number of recipients to trigger the large audience MailTip (Default is 25)", + "placeholder": "Enter a profile name" + } + ], + "label": "Enable all MailTips", "impact": "Low Impact", "impactColour": "info" }, @@ -437,20 +471,29 @@ "impact": "Low Impact", "impactColour": "info" }, + { + "name": "standards.EnableMailboxAuditing", + "cat": "Exchange Standards", + "helpText": "Enables Mailbox auditing for all mailboxes and on tenant level. By default Microsoft does not enable mailbox auditing for Resource Mailboxes, Public Folder Mailboxes and DiscoverySearch Mailboxes. Unified Audit Log needs to be enabled for this standard to function.", + "addedComponent": [], + "label": "Enable Mailbox auditing", + "impact": "Low Impact", + "impactColour": "info" + }, { "name": "standards.SendReceiveLimitTenant", "cat": "Exchange Standards", - "helpText": "Sets the Send and Receive limits for new users. Valid values are 1KB to 150MB. Invalid values will be set to EXO standard of 35MB,36MB", + "helpText": "Sets the Send and Receive limits for new users. Valid values are 1MB to 150MB", "addedComponent": [ { "type": "number", "name": "standards.SendReceiveLimitTenant.SendLimit", - "label": "Send limit in MB" + "label": "Send limit in MB (Default is 35)" }, { "type": "number", "name": "standards.SendReceiveLimitTenant.ReceiveLimit", - "label": "Receive Limit in MB" + "label": "Receive Limit in MB (Default is 36)" } ], "label": "Set send/receive size limits", @@ -507,6 +550,24 @@ "impact": "Low Impact", "impactColour": "info" }, + { + "name": "standards.DisableExternalCalendarSharing", + "cat": "Exchange Standards", + "helpText": "Disables the ability for users to share their calendar with external users. Only for the default policy, so exclusions can be made if needed.", + "addedComponent": [], + "label": "Disable external calendar sharing", + "impact": "Low Impact", + "impactColour": "info" + }, + { + "name": "standards.DisableAdditionalStorageProviders", + "cat": "Exchange Standards", + "helpText": "Disables the ability for users to open files in Outlook on the Web, from other providers such as Box, Dropbox, Facebook, Google Drive, OneDrive Personal, etc.", + "addedComponent": [], + "label": "Disable additional storage providers in OWA", + "impact": "Low Impact", + "impactColour": "info" + }, { "name": "standards.SafeSendersDisable", "cat": "Exchange Standards", @@ -616,7 +677,7 @@ { "name": "standards.DisableAddShortcutsToOneDrive", "cat": "SharePoint Standards", - "helpText": "When the feature is disabled the option Add shortcut to My files will be removed; any folders that have already been added will remain on the user's computer.", + "helpText": "When the feature is disabled the option Add shortcut to OneDrive will be removed. Any folders that have already been added will remain on the user's computer.", "disabledFeatures": { "report": true, "warn": true, diff --git a/src/data/vendorTenantList.json b/src/data/vendorTenantList.json index 5bdebccccde3..de12e3508819 100644 --- a/src/data/vendorTenantList.json +++ b/src/data/vendorTenantList.json @@ -234,5 +234,37 @@ { "vendorName": "SaaSAlerts", "vendorTenantId": "5c7b2b48-9e8f-49ba-80d6-3432e39d596b" + }, + { + "vendorName": "BullPhish (Ciranda)", + "vendorTenantId": "5e4ab895-7a4c-4eea-bb39-75edca0421ad" + }, + { + "vendorName": "Coreview", + "vendorTenantId": "73506dd6-2bc3-49c0-92f3-b2877bab00ba" + }, + { + "vendorName": "Quickpass (CyberQP365)", + "vendorTenantId": "c9006408-eb26-4e50-9bd5-2c078e3dc844" + }, + { + "vendorName": "Infima", + "vendorTenantId": "2bd37396-af18-448c-a391-dd7800364e6f" + }, + { + "vendorName": "Immybot", + "vendorTenantId": "1dcfdedd-ec87-461d-9d55-6989a519d154" + }, + { + "vendorName": "N-Able/Cove backup", + "vendorTenantId": "521c973d-080f-4861-a0cb-8939e59d3d39" + }, + { + "vendorName": "Dropsuite", + "vendorTenantId": "5b8e57d8-5c8e-4b82-98a5-b003bbb26b31" + }, + { + "vendorName": "Huntress", + "vendorTenantId": "19be9add-482a-4c98-ba76-4c2ef7f3bb13" } ] diff --git a/src/routes.js b/src/routes.js index 7b448b3ad39b..2f41ef51dff5 100644 --- a/src/routes.js +++ b/src/routes.js @@ -47,6 +47,7 @@ const GraphExplorer = React.lazy(() => import('src/views/tenant/administration/G const Domains = React.lazy(() => import('src/views/tenant/administration/Domains')) const EditTenant = React.lazy(() => import('src/views/tenant/administration/EditTenant')) const ConditionalAccess = React.lazy(() => import('src/views/tenant/conditional/ConditionalAccess')) +const DeployVacationCA = React.lazy(() => import('src/views/tenant/conditional/DeployVacation')) const NamedLocations = React.lazy(() => import('src/views/tenant/conditional/NamedLocations')) const ListConditionalTemplates = React.lazy(() => @@ -341,6 +342,11 @@ const routes = [ name: 'Conditional Access', component: ConditionalAccess, }, + { + path: '/tenant/conditional/deploy-vacation', + name: 'Deploy Vacation Mode', + component: DeployVacationCA, + }, { path: '/tenant/conditional/list-named-locations', name: 'Named Locations', diff --git a/src/store/middleware/errorMiddleware.js b/src/store/middleware/errorMiddleware.js index 7ad628d6504f..d09c48915ff3 100644 --- a/src/store/middleware/errorMiddleware.js +++ b/src/store/middleware/errorMiddleware.js @@ -17,7 +17,21 @@ export const errorMiddleware = action.payload.data = 'The Azure Function has taken too long to respond. Try selecting a different report or a single tenant instead' } + //if the payload is a string, show the string, if the payload is an object, check if there is a 'Results or 'results' or 'result' property and show that, otherwise show the whole object let message = action.payload?.data || 'A generic error has occurred.' + if (typeof message === 'string') { + // Do nothing, message is already a string + } else if (typeof message === 'object') { + if (message.Results) { + message = message.Results + } else if (message.results) { + message = message.results + } else if (message.result) { + message = message.result + } else { + message = JSON.stringify(message) + } + } if (message.length > 240) { message = message.substring(0, 240) + '...' } diff --git a/src/views/email-exchange/administration/MailboxesList.jsx b/src/views/email-exchange/administration/MailboxesList.jsx index d3a7c92463a7..71bf2ba141b8 100644 --- a/src/views/email-exchange/administration/MailboxesList.jsx +++ b/src/views/email-exchange/administration/MailboxesList.jsx @@ -18,7 +18,7 @@ const MailboxList = () => { return ( <> diff --git a/src/views/endpoint/autopilot/AutopilotListDevices.jsx b/src/views/endpoint/autopilot/AutopilotListDevices.jsx index 92a9ca7382eb..ef74e81d42fc 100644 --- a/src/views/endpoint/autopilot/AutopilotListDevices.jsx +++ b/src/views/endpoint/autopilot/AutopilotListDevices.jsx @@ -1,10 +1,15 @@ -import React from 'react' +import React, { useState } from 'react' import { useSelector } from 'react-redux' import { CButton, CCallout, CSpinner } from '@coreui/react' -import { faTrash } from '@fortawesome/free-solid-svg-icons' +import { + faArrowCircleDown, + faEllipsisV, + faSyncAlt, + faTrash, +} from '@fortawesome/free-solid-svg-icons' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { CippPageList } from 'src/components/layout' -import { ModalService } from 'src/components/utilities' +import { CippActionsOffcanvas, ModalService } from 'src/components/utilities' import { useLazyGenericGetRequestQuery } from 'src/store/api/app' import { CellTip } from 'src/components/tables' import { TitleButton } from 'src/components/buttons' @@ -12,30 +17,66 @@ import { TitleButton } from 'src/components/buttons' const AutopilotListDevices = () => { const tenant = useSelector((state) => state.app.currentTenant) const [ExecuteGetRequest, getResults] = useLazyGenericGetRequestQuery() + const [ocVisible, setOCVisible] = useState(false) + const Actions = (row, index, column) => { - const handleDeleteAPDevice = (apiurl, message) => { - ModalService.confirm({ - title: 'Confirm', - body:
{message}
, - onConfirm: () => ExecuteGetRequest({ path: apiurl }), - confirmLabel: 'Continue', - cancelLabel: 'Cancel', - }) - } return ( - - handleDeleteAPDevice( - `api/RemoveAPDevice?ID=${row.id}&tenantFilter=${tenant.defaultDomainName}`, - 'Do you want to delete the Autopilot Device?', - ) - } - > - - + <> + setOCVisible(true)}> + + + setOCVisible(false)} + /> + ) } @@ -102,9 +143,20 @@ const AutopilotListDevices = () => { +
- +
+ + ExecuteGetRequest({ + path: `/api/ExecSyncAPDevices?tenantFilter=${tenant.defaultDomainName}`, + }) + } + title="Sync Devices" + /> +
+
} datatable={{ keyField: 'id', diff --git a/src/views/identity/administration/AddUser.jsx b/src/views/identity/administration/AddUser.jsx index ba0e0b1d96a3..a915ec793bd0 100644 --- a/src/views/identity/administration/AddUser.jsx +++ b/src/views/identity/administration/AddUser.jsx @@ -1,4 +1,4 @@ -import React from 'react' +import React, { useEffect, useState } from 'react' import { CButton, CCard, @@ -36,6 +36,7 @@ import { required } from 'src/validators' import useQuery from 'src/hooks/useQuery' import Select from 'react-select' import { useNavigate } from 'react-router-dom' +import { OnChange } from 'react-final-form-listeners' const AddUser = () => { let navigate = useNavigate() @@ -103,11 +104,7 @@ const AddUser = () => { genericPostRequest({ path: '/api/AddUser', values: shippedValues }) } const usagelocation = useSelector((state) => state.app.usageLocation) - const initialState = { - Autopassword: false, - usageLocation: usagelocation, - ...allQueryObj, - } + const copyUserVariables = (t) => { for (const [key, value] of Object.entries(t.value)) { query.delete(key) @@ -117,6 +114,20 @@ const AddUser = () => { navigate(`?${query.toString()}`) } } + + const [firstName, setFirstName] = useState('') + const [lastName, setLastName] = useState('') + const [displayName, setDisplayName] = useState('') + const initialState = { + Autopassword: false, + usageLocation: usagelocation, + ...allQueryObj, + } + // Effect to update display name when first or last name changes + useEffect(() => { + setDisplayName(`${firstName} ${lastName}`) + }, [firstName, lastName, displayName]) + return ( {postResults.isSuccess && ( @@ -144,15 +155,25 @@ const AddUser = () => {
{ + render={({ form, handleSubmit, submitting, values }) => { return ( - + setFirstName(e.target.value)} + /> - + setLastName(e.target.value)} + /> @@ -163,6 +184,16 @@ const AddUser = () => { label="Display Name" validate={required} /> + + {(value) => { + form.change('displayName', `${value} ${lastName}`) + }} + + + {(value) => { + form.change('displayName', `${firstName} ${value} `) + }} + diff --git a/src/views/identity/administration/Users.jsx b/src/views/identity/administration/Users.jsx index 360693314d62..f782b436c629 100644 --- a/src/views/identity/administration/Users.jsx +++ b/src/views/identity/administration/Users.jsx @@ -88,11 +88,6 @@ const Offcanvas = (row, rowIndex, formatExtraData) => { link: `/identity/administration/ViewBec?userId=${row.id}&tenantDomain=${tenant.defaultDomainName}&ID=${row.userPrincipalName}`, color: 'info', }, - { - label: 'Offboard User', - link: OffboardLink, - color: 'info', - }, { label: 'Create Temporary Access Password', color: 'info', @@ -140,6 +135,54 @@ const Offcanvas = (row, rowIndex, formatExtraData) => { }, modalMessage: 'Select the sharepoint site to create a shortcut for', }, + { + label: 'Add to group', + color: 'info', + modal: true, + modalType: 'POST', + modalBody: { + Addmember: { + value: row.userPrincipalName, + }, + TenantId: tenant.defaultDomainName, + }, + modalUrl: `/api/EditGroup`, + modalDropdown: { + url: `/api/listGroups?TenantFilter=${tenant.defaultDomainName}`, + labelField: 'displayName', + valueField: 'id', + addedField: { + groupId: 'id', + groupType: 'calculatedGroupType', + groupName: 'displayName', + }, + }, + modalMessage: 'Select the group to add the user to', + }, + { + label: 'Remove from group', + color: 'info', + modal: true, + modalType: 'POST', + modalBody: { + Removemember: { + value: row.userPrincipalName, + }, + TenantId: tenant.defaultDomainName, + }, + modalUrl: `/api/EditGroup`, + modalDropdown: { + url: `/api/listGroups?TenantFilter=${tenant.defaultDomainName}`, + labelField: 'displayName', + valueField: 'id', + addedField: { + groupId: 'id', + groupType: 'calculatedGroupType', + groupName: 'displayName', + }, + }, + modalMessage: 'Select the group to add the user to', + }, { label: 'Enable Online Archive', color: 'info', @@ -217,13 +260,6 @@ const Offcanvas = (row, rowIndex, formatExtraData) => { modalUrl: `/api/ExecResetPass?MustChange=false&TenantFilter=${tenant.defaultDomainName}&ID=${row.id}&displayName=${row.displayName}`, modalMessage: 'Are you sure you want to reset the password for this user?', }, - { - label: 'Clear ImmutableId', - color: 'warning', - modal: true, - modalUrl: `/api/ExecClrImmId?TenantFilter=${tenant.defaultDomainName}&ID=${row.id}`, - modalMessage: 'Are you sure you want to clear the ImmutableId for this user?', - }, { label: 'Revoke all user sessions', color: 'danger', @@ -449,7 +485,7 @@ const Users = (row) => { label: 'Enable Online Archive', color: 'info', modal: true, - modalUrl: `/api/ExecEnableArchive?TenantFilter=!Tenant&ID=!id`, + modalUrl: `/api/ExecEnableArchive?TenantFilter=!Tenant&ID=!userPrincipalName`, modalMessage: 'Are you sure you want to enable the online archive for these users?', }, { diff --git a/src/views/tenant/administration/GeoIPLookup.jsx b/src/views/tenant/administration/GeoIPLookup.jsx index dd1b21e215a4..339a93630a99 100644 --- a/src/views/tenant/administration/GeoIPLookup.jsx +++ b/src/views/tenant/administration/GeoIPLookup.jsx @@ -1,6 +1,7 @@ import React, { useEffect, useState } from 'react' import { CButton, + CCallout, CCard, CCardBody, CCardHeader, @@ -57,6 +58,18 @@ const GeoIPLookup = () => { }) } }, [execGraphRequest, tenant.defaultDomainName, query, ip]) + const [execAddIp, iprequest] = useLazyGenericGetRequestQuery() + + const addTrustedIP = (State) => { + execAddIp({ + path: 'api/ExecAddTrustedIP', + params: { + IP: ip, + TenantFilter: tenant.defaultDomainName, + State: State, + }, + }) + } return ( @@ -115,28 +128,71 @@ const GeoIPLookup = () => { {ip} -

Range

+

AS

{graphrequest.isFetching && } - {graphrequest.data?.startaddress} - {graphrequest.data?.endAddress} + {graphrequest.data?.as}

Owner

{graphrequest.isFetching && } - {graphrequest.data?.OrgRef} + {graphrequest.data?.org}
-

Subnet Name

+

ISP

{graphrequest.isFetching && } - {graphrequest.data?.SubnetName} + {graphrequest.data?.isp}
- +

Geo IP Location

{graphrequest.isFetching && } - {graphrequest.data?.location?.countryCode} - {graphrequest.data?.location?.cityName} + {graphrequest.data?.country} - {graphrequest.data?.city} +
+ +

Lat/Lon

+ {graphrequest.isFetching && } + {graphrequest.data?.lat} / {graphrequest.data?.lon} +
+
+ + +

Hosting

+ {graphrequest.isFetching && } + {graphrequest.data?.hosting ? 'Yes' : 'No'} +
+ +

Mobile

+ {graphrequest.isFetching && } + {graphrequest.data?.mobile ? 'Yes' : 'No'} +
+ +

Proxy or Anonimizer

+ {graphrequest.isFetching && } + {graphrequest.data?.proxy ? 'Yes' : 'No'} +
+
+ + + addTrustedIP('Trusted')} className="me-3"> + Add as trusted IP for selected tenant + {iprequest.isFetching && } + + addTrustedIP('NotTrusted')} + > + Remove as trusted IP for selected tenant + {iprequest.isFetching && } + + {iprequest.data && ( + + {iprequest.data?.results} + + )} )} diff --git a/src/views/tenant/administration/TenantOffboardingWizard.jsx b/src/views/tenant/administration/TenantOffboardingWizard.jsx index 38baf73903d8..81749ba3221b 100644 --- a/src/views/tenant/administration/TenantOffboardingWizard.jsx +++ b/src/views/tenant/administration/TenantOffboardingWizard.jsx @@ -222,7 +222,7 @@ const TenantOffboardingWizard = () => { diff --git a/src/views/tenant/conditional/DeployVacation.jsx b/src/views/tenant/conditional/DeployVacation.jsx new file mode 100644 index 000000000000..8f1fb21b7281 --- /dev/null +++ b/src/views/tenant/conditional/DeployVacation.jsx @@ -0,0 +1,184 @@ +import React, { useState } from 'react' +import { CButton, CCallout, CCol, CForm, CRow, CSpinner, CTooltip } from '@coreui/react' +import { useSelector } from 'react-redux' +import { Field, Form } from 'react-final-form' +import { Condition, RFFCFormSwitch, RFFSelectSearch } from 'src/components/forms' +import { + useGenericGetRequestQuery, + useLazyGenericGetRequestQuery, + useLazyGenericPostRequestQuery, +} from 'src/store/api/app' +import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' +import { faCircleNotch, faEdit, faEye } from '@fortawesome/free-solid-svg-icons' +import { CippContentCard, CippPage, CippPageList } from 'src/components/layout' +import { CellTip } from 'src/components/tables/CellGenericFormat' +import 'react-datepicker/dist/react-datepicker.css' +import { CippActionsOffcanvas, ModalService, TenantSelector } from 'src/components/utilities' +import arrayMutators from 'final-form-arrays' +import DatePicker from 'react-datepicker' +import 'react-datepicker/dist/react-datepicker.css' +import { useListUsersQuery } from 'src/store/api/users' +import { useListConditionalAccessPoliciesQuery } from 'src/store/api/tenants' + +const ListClassicAlerts = () => { + const [ExecuteGetRequest, getResults] = useLazyGenericGetRequestQuery() + const currentDate = new Date() + const [startDate, setStartDate] = useState(currentDate) + const [endDate, setEndDate] = useState(currentDate) + + const tenantDomain = useSelector((state) => state.app.currentTenant.defaultDomainName) + const [refreshState, setRefreshState] = useState(false) + const [genericPostRequest, postResults] = useLazyGenericPostRequestQuery() + + const onSubmit = (values) => { + const startTime = Math.floor(startDate.getTime() / 1000) + const endTime = Math.floor(endDate.getTime() / 1000) + const shippedValues = { + tenantFilter: tenantDomain, + UserId: values.UserId?.value, + PolicyId: values.PolicyId?.value, + StartDate: startTime, + EndDate: endTime, + vacation: true, + } + genericPostRequest({ path: '/api/ExecCAExclusion', values: shippedValues }).then((res) => { + setRefreshState(res.requestId) + }) + } + + const { + data: users = [], + isFetching: usersIsFetching, + error: usersError, + } = useListUsersQuery({ tenantDomain }) + + const { + data: caPolicies = [], + isFetching: caIsFetching, + error: caError, + } = useListConditionalAccessPoliciesQuery({ domain: tenantDomain }) + + return ( + + <> + + + + { + return ( + +

+ Vacation mode adds a scheduled tasks to add and remove users from CA + exclusions for a specific period of time. Select the CA policy and the date + range. +

+ + + + {(props) => } + + + +
+
+ + + ({ + value: user.id, + name: `${user.displayName} <${user.userPrincipalName}>`, + }))} + placeholder={!usersIsFetching ? 'Select user' : 'Loading...'} + name="UserId" + /> + + + + + ({ + value: ca.id, + name: `${ca.displayName}`, + }))} + placeholder={!caIsFetching ? 'Select user' : 'Loading...'} + name="PolicyId" + /> + + + + + setStartDate(date)} + /> + + + + setEndDate(date)} + /> + + + + + Set Vacation Mode + {postResults.isFetching && ( + + )} + + + + {postResults.isSuccess && ( + +
  • {postResults.data.Results}
  • +
    + )} + {getResults.isFetching && ( + + Loading + + )} + {getResults.isSuccess && ( + {getResults.data?.Results} + )} + {getResults.isError && ( + + Could not connect to API: {getResults.error.message} + + )} +
    + ) + }} + /> +
    +
    +
    + +
    + ) +} + +export default ListClassicAlerts diff --git a/src/views/tenant/standards/BestPracticeAnalyser.jsx b/src/views/tenant/standards/BestPracticeAnalyser.jsx index 1d05c0be6278..3c94d28db3fd 100644 --- a/src/views/tenant/standards/BestPracticeAnalyser.jsx +++ b/src/views/tenant/standards/BestPracticeAnalyser.jsx @@ -40,6 +40,7 @@ import { CellTip, cellGenericFormatter } from 'src/components/tables/CellGeneric import { useExecBestPracticeAnalyserMutation } from 'src/store/api/reports' import { ModalService } from 'src/components/utilities' import { cellTableFormatter } from 'src/components/tables/CellTable' +import { cellMathFormatter } from 'src/components/tables/CellMathFormatter' const RefreshAction = ({ singleTenant = false, refreshFunction = null }) => { const tenantDomain = useSelector((state) => state.app.currentTenant.defaultDomainName) @@ -187,6 +188,9 @@ const BestPracticeAnalyser = () => { case 'table': cellSelector = cellTableFormatter(col.value) break + case 'math': + cellSelector = cellMathFormatter({ col }) + break default: cellSelector = cellGenericFormatter() break diff --git a/src/views/tenant/standards/DomainsAnalyser.jsx b/src/views/tenant/standards/DomainsAnalyser.jsx index dbe792deaf7e..e7cd129eaa14 100644 --- a/src/views/tenant/standards/DomainsAnalyser.jsx +++ b/src/views/tenant/standards/DomainsAnalyser.jsx @@ -285,7 +285,7 @@ const DomainsAnalyser = () => { filter: 'Complex: domain notlike onmicrosoft', }, ], - path: `/api/DomainAnalyser_List`, + path: `/api/ListDomainAnalyser`, params: { tenantFilter: currentTenant.defaultDomainName }, columns, reportName: 'Domains-Analyzer', diff --git a/version_latest.txt b/version_latest.txt index 1f1ac7c2f330..28cbf7c0aae3 100644 --- a/version_latest.txt +++ b/version_latest.txt @@ -1 +1 @@ -4.9.1 \ No newline at end of file +5.0.0 \ No newline at end of file