Query Details

Multiple URL Entity Email Url Info

Query

// This query assumes a feed of threat indicators is ingested/synchronized periodically, and each synchronization ingests new indicators and only old indicators that have been modified.
// Active threat indicators in Sentinel are renovated as ThreatIntelligenceIndicator events every ~12 days.
let query_frequency = 1h;
let query_period = 14d;
let query_wait = 1h;
let table_query_lookback = 14d;
let _TIBenignProperty =
    _GetWatchlist('ID-TIBenignProperty')
    | where Notes has_any ("[URL]")
    | project IndicatorId, BenignProperty
;
let _TIExcludedSources = toscalar(
    _GetWatchlist('Activity-ExpectedSignificantActivity')
    | where Activity == "ThreatIndicatorSource"
    | summarize make_list(Auxiliar)
    );
let _URLRegex = toscalar(
    _GetWatchlist('RegEx-SingleRegularExpressions')
    | where UseCase == "Threat Intelligence Indicator URL"
    | project RegEx
    );
let _ExpectedEmails =
    _GetWatchlist('Activity-ExpectedSignificantActivity')
    | where Activity == "MaliciousURLSentEmail"
    | project
        SenderFromAddress = SourceAddress,
        RecipientEmailAddress = DestinationAddress,
        Subject = Auxiliar
;
let _TITableMatch = (table_start: datetime, table_end: datetime, only_new_ti: boolean, ti_start: datetime = datetime(null)) {
    // Scheduled Analytics rules have a query period limit of 14d
    let _Indicators =// materialize(
        ThreatIntelligenceIndicator
        | where TimeGenerated > ago(query_period)
        // Take the earliest TimeGenerated and the latest column info
        | summarize hint.strategy=shuffle
            minTimeGenerated = min(TimeGenerated),
            arg_max(TimeGenerated, Active, Description, ActivityGroupNames, IndicatorId, ThreatType, DomainName, Url, ExpirationDateTime, ConfidenceScore, AdditionalInformation, ExternalIndicatorId)
            by IndicatorId
        // Remove inactive or expired indicators
        | where not(not(Active) or ExpirationDateTime < now())
        // Pick indicators that contain the desired entity type
        | where isnotempty(Url)
        | extend Url = trim_end(@"\/", Url)
        // Remove indicators from specific sources
        | where not(AdditionalInformation has_any (_TIExcludedSources))
        // Remove excluded indicators with benign properties
        | join kind=leftanti _TIBenignProperty on IndicatorId, $left.Url == $right.BenignProperty
        // Deduplicate indicators by Url column, equivalent to using join kind=innerunique afterwards
        | summarize hint.strategy=shuffle
            minTimeGenerated = min(minTimeGenerated),
            take_any(*)
            by Url
        // If we want only new indicators, remove indicators received previously
        | where not(only_new_ti and minTimeGenerated < ti_start)
    //)
    ;
    //let _IndicatorsLength = toscalar(_Indicators | summarize count());
    //let _IndicatorsPrefilter = toscalar(
    //    _Indicators
    //    | extend AuxiliarField = tostring(split(extract(_URLRegex, 3, Url), ".")[-1])
    //    | summarize make_set_if(AuxiliarField, isnotempty(AuxiliarField), 10000)
    //);
    //let _IndicatorsPrefilterLength = array_length(_IndicatorsPrefilter);
    let _TableEvents =
        EmailUrlInfo
        | where ingestion_time() between (table_start .. table_end)
        // Filter events that may contain indicators
        | where isnotempty(Url)
        //| where not(_IndicatorsPrefilterLength < 10000 and not(Url has_any (_IndicatorsPrefilter))) // valid TLD ~1500 , "has_any" limit 10000
        | extend Url = trim_end(@"\/", Url)
        //| where not(_IndicatorsLength < 1000000 and not(Url in (toscalar(_Indicators | summarize make_list(Url))))) // "in" limit 1.000.000
        | extend EmailUrlInfo_Url = Url
        | project-rename EmailUrlInfo_TimeGenerated = TimeGenerated
    ;
    _Indicators
    | join kind=inner hint.strategy=shuffle _TableEvents on Url
    // Take only a single event by key columns
    //| summarize hint.strategy=shuffle take_any(*) by Url, NetworkMessageId
    | project
        EmailUrlInfo_TimeGenerated,
        Description, ActivityGroupNames, IndicatorId, ThreatType, DomainName, Url, ExpirationDateTime, ConfidenceScore, AdditionalInformation,
        UrlLocation, EmailUrlInfo_Url, UrlDomain, NetworkMessageId, ReportId
    | join kind=leftouter hint.strategy=shuffle (
        EmailEvents
        | where ingestion_time() between(table_start .. now())
        | project
            SenderFromAddress,
            SenderFromDomain,
            SenderMailFromAddress,
            SenderMailFromDomain,
            SenderDisplayName,
            SenderIPv4,
            SenderIPv6,
            AuthenticationDetails,
            RecipientEmailAddress,
            EmailDirection,
            Subject,
            EmailLanguage,
            UrlCount,
            AttachmentCount,
            AdditionalFields,
            OrgLevelPolicy,
            OrgLevelAction,
            UserLevelPolicy,
            UserLevelAction,
            EmailActionPolicy,
            EmailAction,
            DeliveryAction,
            DeliveryLocation,
            ThreatTypes,
            ConfidenceLevel,
            DetectionMethods,
            Connectors,
            NetworkMessageId,
            EmailEvents_ReportId = ReportId
        )
        on NetworkMessageId
    | project-away NetworkMessageId1
};
union// isfuzzy=true
    // Match      current table events                                all indicators available
    _TITableMatch(ago(query_frequency + query_wait), ago(query_wait), false),
    // Match      past table events                                                          new indicators since last query execution
    _TITableMatch(ago(table_query_lookback + query_wait), ago(query_frequency + query_wait), true, ago(query_frequency))
| summarize arg_max(EmailUrlInfo_TimeGenerated, *) by IndicatorId, NetworkMessageId
| extend
    timestamp = EmailUrlInfo_TimeGenerated,
    URLCustomEntity = Url
| join kind=leftanti _ExpectedEmails on SenderFromAddress

Explanation

This query is used to match threat indicators with table events in order to identify any potential threats. The query assumes that threat indicators are ingested periodically and only new or modified indicators are ingested. It also sets some variables for the query frequency, period, and wait time.

The query first retrieves a list of benign properties and excluded sources from watchlists. It then retrieves a regular expression for threat intelligence indicator URLs and a list of expected emails.

The main part of the query is the _TITableMatch function, which matches table events with threat indicators. It filters out inactive or expired indicators, removes indicators from specific sources, and removes indicators with benign properties. It also deduplicates indicators based on the URL column. If the only_new_ti parameter is set to true, it removes indicators received before a specified start time.

The query then joins the threat indicators with table events and email events. It takes only a single event for each key column and projects relevant columns for further analysis. It also joins with the list of expected emails to remove any false positives.

Finally, the query combines the results from two _TITableMatch function calls and summarizes the data by IndicatorId and NetworkMessageId. It adds a timestamp and URL custom entity column and removes any matches with expected emails.

Details

Jose Sebastián Canós profile picture

Jose Sebastián Canós

Released: December 5, 2022

Tables

ThreatIntelligenceIndicatorEmailUrlInfoEmailEvents

Keywords

Devices,Intune,User

Operators

letwherehas_anyprojecttoscalarsummarizemake_listjoinextendtrim_endnotisnotemptyisnotemptyisnotemptyhint.strategy=shufflebyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotemptyisnotempty

Actions