From 2aae720d43b03e6efb6c40c7f265cea31c5924c9 Mon Sep 17 00:00:00 2001 From: Vedika Gupta Date: Fri, 30 Jan 2026 17:25:23 +0530 Subject: [PATCH 1/7] topic changes for get full profile --- .../WorkdayGetUserProfile/topic.yaml | 25 +++++++++++++------ 1 file changed, 18 insertions(+), 7 deletions(-) diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml index 79909aea..793f26fc 100644 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml @@ -1,15 +1,26 @@ kind: AdaptiveDialog modelDescription: |- - Respond to user's profile information requests from Workday. Provide data ONLY for the requesting user, never for others (managers, colleagues, etc.). + Respond to profile requests from Workday. Data is ONLY for the requesting user. - Available fields: Name, Employee ID, DOB, Gender, Business Title, Organization (Department), Manager, Location, Hire Date, Work Email, Work Phone, Home Email, Home Phone, Home Address, Employment Status. + Start with: "Sure, here's your employee profile from [Workday](https://www.workday.com), your company's HR platform." - Answer based on which specific data the user asks for. Include relevant fields based on the question. + Format with section headers (not bold labels). Structure: + + Personal information: Name, Employee ID + Hiring details: Hire date, Length of service (only if asked about tenure) + Role and work: Position, Manager, Location + Team and organization: Department + Contact information: Work email, Work phone, Personal email, Personal phone, Home address - Include Continuous Service Date and Length of Service ONLY when user specifically asks about tenure, service length, years of service, or how long they've been with the company. + Rules: + - Use section headers, bullet lists within sections + - Do NOT bold every label + - Home address must be a single line, NOT sub-bullets + - Exclude DOB/Gender unless specifically asked + - For specific questions, answer directly without all sections + - End with: "Let me know if you want details about a section or to update your profile." - Invalid requests: "What is my manager's job title?" "What is my colleague's employee ID?" "What department is John in?" - Valid requests: "What is my employee ID?" "What is my job title?" "Show my profile" "What is my work email?" "What is my tenure?" (include service length) + Invalid: other people's data. Valid: own profile/email/title/tenure. beginDialog: kind: OnRecognizedIntent @@ -185,7 +196,7 @@ beginDialog: Location: First(Topic.parsedWorkdayResponse.Location).Value, HireDate: First(Topic.parsedWorkdayResponse.HireDate).Value, WorkEmail: First(Topic.parsedWorkdayResponse.WorkEmail).Value, - HomeAddress: First(Topic.parsedWorkdayResponse.HomeAddress).Value, + HomeAddress: Substitute(Substitute(Concat(Topic.parsedWorkdayResponse.HomeAddress, Value, ", "), Char(10), ", "), Char(13), ""), HomeEmail: First(Topic.parsedWorkdayResponse.HomeEmail).Value, HomePhone: First(Topic.parsedWorkdayResponse.HomePhone).Value, WorkPhone: First(Topic.parsedWorkdayResponse.WorkPhone).Value, From 3c0db3bd1842857bb8675e5bc37d3267b5b8d0dc Mon Sep 17 00:00:00 2001 From: Vedika Gupta Date: Fri, 30 Jan 2026 19:08:33 +0530 Subject: [PATCH 2/7] reportees time in position and get user profile --- .../WorkdayGetUserProfile/topic.yaml | 26 ++++++++----------- .../topic.yaml | 12 ++++----- 2 files changed, 17 insertions(+), 21 deletions(-) diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml index 793f26fc..8b1c78c3 100644 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml @@ -1,26 +1,22 @@ kind: AdaptiveDialog modelDescription: |- - Respond to profile requests from Workday. Data is ONLY for the requesting user. + Respond to profile requests from [Workday](https://www.workday.com), an HR platform your company uses. - Start with: "Sure, here's your employee profile from [Workday](https://www.workday.com), your company's HR platform." + For FULL PROFILE: Start with "Sure, here's your employee profile from Workday..." then use sections: + Personal information, Hiring details, Role and work, Team and organization, Contact information. - Format with section headers (not bold labels). Structure: - - Personal information: Name, Employee ID - Hiring details: Hire date, Length of service (only if asked about tenure) - Role and work: Position, Manager, Location - Team and organization: Department - Contact information: Work email, Work phone, Personal email, Personal phone, Home address + For TENURE/POSITION questions: Start with "Here's how long you've been in your position according to Workday..." + Include role title: "You've been in your current role as a [Position] for [LengthOfService]." + Add "About your position" section: Department, Location. Rules: + - Always include Workday attribution - Use section headers, bullet lists within sections - - Do NOT bold every label - - Home address must be a single line, NOT sub-bullets - - Exclude DOB/Gender unless specifically asked - - For specific questions, answer directly without all sections - - End with: "Let me know if you want details about a section or to update your profile." + - Home address as single line, NOT sub-bullets + - Exclude DOB/Gender unless asked + - End with: "Let me know if you want more details or to update your profile." - Invalid: other people's data. Valid: own profile/email/title/tenure. + Invalid: other people's data. beginDialog: kind: OnRecognizedIntent diff --git a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml index adecac4b..90453187 100644 --- a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml @@ -1,14 +1,14 @@ kind: AdaptiveDialog modelDescription: |- - You will respond to requests about time in position for direct reports of the user making the request. There is no information available for anyone who isn't a direct report, and being a direct report means that the individual is not a manager, spouse, sibling, or any other relationship to the requestor, and only means that they report to the requestor. The resulting data will contain a list of employees who report to the requestor and will contain their position start date, time in position, business title, job profile, location, hire date, and status. You must NOT give data if you do not have enough info. + Respond to team time-in-position requests from [Workday](https://www.workday.com), an HR platform your company uses. + For multiple team members: Use a simple table with Name, Title, Time in position. + For one team member: Use their name as header with bullet list of details. - Example valid request: - "What is the time in position of my direct reports?" - "Show my team's time in position" - "How long have my reportees been in their current positions?" + Keep responses concise. Do NOT show Status or Hire Date unless asked. + End with: "Let me know if you have questions about your team." - Your output **must** be a nested list in markdown language based on the data contained in the {Topic.workdayResponseTableWithTimeInPosition} variable + Invalid: non-direct reports. beginDialog: kind: OnRecognizedIntent id: main From 1159ec1f494c07b8c55e5ac2ba31cdfdc8484c31 Mon Sep 17 00:00:00 2001 From: Ankur Rana Date: Fri, 30 Jan 2026 21:06:08 +0530 Subject: [PATCH 3/7] Add WorkdayEmployeeRequestTimeOff-V2 with multi-day date range support --- .../RequestTimeOff_Topic.yaml | 741 ++++++++++++++++++ ..._HRWorkdayAbsenceEnterTimeOff_MultiDay.xml | 53 ++ 2 files changed, 794 insertions(+) create mode 100644 EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml create mode 100644 EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml new file mode 100644 index 00000000..7b5849b9 --- /dev/null +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml @@ -0,0 +1,741 @@ +kind: AdaptiveDialog +inputs: + - kind: AutomaticTaskInput + propertyName: EmployeeName + description: The name of the employee to fetch data for. + entity: PersonNamePrebuiltEntity + shouldPromptUser: false + + - kind: AutomaticTaskInput + propertyName: InputStartDate + description: The start date of the time off request. + entity: DateTimePrebuiltEntity + shouldPromptUser: false + + - kind: AutomaticTaskInput + propertyName: InputEndDate + description: The end date of the time off request. + entity: DateTimePrebuiltEntity + shouldPromptUser: false + + - kind: AutomaticTaskInput + propertyName: InputHoursPerDay + description: The number of hours per day for the time off request. + entity: NumberPrebuiltEntity + shouldPromptUser: false + + - kind: AutomaticTaskInput + propertyName: InputTimeOffType + description: Extract the type of leave mentioned such as vacation, sick, sick leave, floating holiday, floater, PTO, annual leave, medical, or illness. Extract keywords like 'sick', 'vacation', 'floating', 'PTO', 'annual'. + entity: StringPrebuiltEntity + shouldPromptUser: false + +modelDescription: |- + You will respond only to requests related to requesting time off for the user making the request. + All time off requests are submitted to Workday (Absence_Management) and pertain exclusively to the requesting user. + This topic may prompt the user for input (time off type, start date, end date, and reason) if they are requesting time off for themselves. + This data exclusively belongs to the user making the request. Do not respond to questions about other people's data. + + Example invalid requests: + "Request time off for my manager" + "Request time off for my sister" + "Request time off for [EmployeeName]" + + Example valid requests: + "Request time off" + "I need to submit time off" + "Please request vacation from January 5th to January 10th" + "Request sick leave for next week" + "I need time off from 2025-09-15 to 2025-09-20 for a family event" +beginDialog: + kind: OnRecognizedIntent + id: main + intent: {} + actions: + - kind: SetVariable + id: set_workday_url + variable: Topic.WorkdayUrl + value: https://impl.workday.com/microsoft_dpt6/d/home.htmld + + - kind: SetVariable + id: set_workday_icon_url + variable: Topic.WorkdayIconUrl + value: https://conn-afd-prod-endpoint-bmc9bqahasf3grgk.b01.azurefd.net/v1.0.1774/1.0.1774.4397/workdaysoap/icon.png + + # ================================================================ + # DATE CONFIGURATION + # Set the effective date for balance lookup. Default is Today(). + # Customers can change this to a specific date if needed. + # ================================================================ + - kind: SetVariable + id: set_as_of_effective_date + variable: Topic.AsOfEffectiveDate + value: =Today() + + # ================================================================ + # LEAVE TYPE CONFIGURATION + # Edit keywords below to customize how user phrases map to leave types. + # Keywords are comma-separated and case-insensitive. + # LeaveTypeValue must match the dropdown choice values. + # ================================================================ + - kind: SetVariable + id: set_leave_type_config + variable: Topic.LeaveTypeConfig + value: |- + =Table( + {Keywords: "vacation,annual,pto,holiday pay", LeaveTypeValue: "Vacation_Hours"}, + {Keywords: "floating,floater,float day", LeaveTypeValue: "Floating_Holiday_Hours"}, + {Keywords: "sick,illness,medical,unwell,not feeling well", LeaveTypeValue: "Sick_Hours"} + ) + + # Map extracted time off type to dropdown value using config table + - kind: SetVariable + id: set_mapped_time_off_type + variable: Topic.MappedTimeOffType + value: |- + =If( + IsBlank(Topic.InputTimeOffType), + "", + Coalesce( + First( + Filter( + Topic.LeaveTypeConfig, + !IsEmpty(Filter(Split(Keywords, ","), Trim(Value) in Lower(Topic.InputTimeOffType))) + ) + ).LeaveTypeValue, + "" + ) + ) + + - kind: BeginDialog + id: get_leave_balance + displayName: Get Leave Balance + input: + binding: + parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{As_Of_Effective_Date}"",""value"":""" & Text(Topic.AsOfEffectiveDate, "yyyy-MM-dd") & """}]}" + scenarioName: msdyn_HRWorkdayHCMEmployeeGetVacationBalance + + dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution + output: + binding: + errorResponse: Topic.balanceErrorResponse + isSuccess: Topic.balanceIsSuccess + workdayResponse: Topic.balanceResponse + + - kind: ParseValue + id: parse_balance + variable: Topic.parsedBalance + valueType: + kind: Record + properties: + PlanDescriptor: + type: + kind: Table + properties: + Value: String + + PlanID: + type: + kind: Table + properties: + Value: String + + RemainingBalance: + type: + kind: Table + properties: + Value: String + + UnitOfTime: + type: + kind: Table + properties: + Value: String + + value: =Topic.balanceResponse + + # ================================================================ + # PLAN BALANCE CONFIGURATION + # Maps Plan IDs (from balance API) to display names. + # Update PlanID values to match your Workday configuration. + # Only plans listed here AND returned by the API will be displayed. + # ================================================================ + - kind: SetVariable + id: set_plan_config + variable: Topic.PlanConfig + value: |- + =Table( + {PlanID: "PTO_USA", DisplayName: "Paid time off"}, + {PlanID: "FH_USA", DisplayName: "Floating holiday"}, + {PlanID: "ABSENCE_PLAN-6-159", DisplayName: "Sick leave"}, + {PlanID: "ABSENCE_PLAN-6-158", DisplayName: "Vacation"} + ) + + # Create merged balance table for easy lookup by PlanID + - kind: SetVariable + id: set_balance_table + variable: Topic.BalanceTable + value: |- + =ForAll( + Sequence(CountRows(Topic.parsedBalance.PlanID)), + { + PlanID: Index(Topic.parsedBalance.PlanID, Value).Value, + Balance: Index(Topic.parsedBalance.RemainingBalance, Value).Value + } + ) + + # Build display data: only include plans that exist in BOTH config AND API response + - kind: SetVariable + id: set_display_balances + variable: Topic.DisplayBalances + value: |- + =ForAll( + Filter( + Topic.PlanConfig, + !IsBlank(LookUp(Topic.BalanceTable, PlanID = ThisRecord.PlanID).Balance) + ) As plan, + { + PlanID: plan.PlanID, + DisplayName: plan.DisplayName, + Balance: LookUp(Topic.BalanceTable, PlanID = plan.PlanID).Balance + } + ) + + - kind: SetVariable + id: build_intro_message + variable: Topic.introMessage + value: ="Sure, I'll help you submit a time off request. Here's a form where you can choose the type of leave and dates you want off. Don't see the type of leave you want? [Book directly in Workday](" & Topic.WorkdayUrl & ")" + + - kind: SendActivity + id: intro_message + activity: "{Topic.introMessage}" + + - kind: ConditionGroup + id: need_inputs + conditions: + - id: always_true + condition: true + actions: + - kind: AdaptiveCardPrompt + id: collect_time_off + displayName: Ask time off type, start date, end date and reason + card: |- + ={ + type: "AdaptiveCard", + '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", + version: "1.5", + body: [ + { + type: "TextBlock", + text: "Book your time off", + weight: "Bolder", + size: "Medium", + wrap: true, + color: "Default" + }, + { + type: "Container", + style: "emphasis", + bleed: true, + items: [ + { + type: "TextBlock", + text: "Available balance in hours as on " & Text(Topic.AsOfEffectiveDate, "mmmm d, yyyy"), + size: "Small", + weight: "Bolder", + wrap: true + }, + { + type: "ColumnSet", + columns: ForAll( + Topic.DisplayBalances, + { + type: "Column", + width: "stretch", + items: [ + { type: "TextBlock", text: DisplayName, size: "Small", wrap: true }, + { type: "TextBlock", text: Balance, weight: "Bolder", spacing: "Small" } + ] + } + ) + } + ] + }, + { + type: "TextBlock", + text: "Fields marked with * are required.", + wrap: true, + spacing: "Medium", + size: "Small" + }, + { + type: "ColumnSet", + columns: [ + { + type: "Column", + width: "stretch", + items: [ + { + type: "Input.ChoiceSet", + id: "timeOffType", + label: "Type of time off", + style: "compact", + value: Topic.MappedTimeOffType, + isRequired: true, + errorMessage: "Please select a type.", + choices: [ + { title: "Vacation", value: "Vacation_Hours" }, + { title: "Floating holiday", value: "Floating_Holiday_Hours" }, + { title: "Sick leave", value: "Sick_Hours" }, + { title: "I don't see my leave type listed", value: "NOT_LISTED" } + ] + } + ] + } + ] + }, + { + type: "Input.Date", + id: "startDate", + label: "Start date", + value: If(IsBlank(Topic.InputStartDate), "", Text(Topic.InputStartDate, "yyyy-MM-dd")), + isRequired: true, + errorMessage: "Please select a start date." + }, + { + type: "Input.Date", + id: "endDate", + label: "End date", + value: If(IsBlank(Topic.InputEndDate), "", Text(Topic.InputEndDate, "yyyy-MM-dd")), + isRequired: true, + errorMessage: "Please select an end date." + }, + { + type: "Input.Number", + id: "hoursPerDay", + label: "Hours per day", + min: 1, + max: 24, + value: If(IsBlank(Topic.InputHoursPerDay), 8, Value(Topic.InputHoursPerDay)), + isRequired: true, + errorMessage: "Please enter hours per day." + }, + { + type: "ColumnSet", + spacing: "Medium", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "ActionSet", + actions: [ + { type: "Action.Submit", title: "Submit" } + ] + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "stretch", + items: [], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "ColumnSet", + spacing: "None", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "Image", + url: Topic.WorkdayIconUrl, + size: "Small", + height: "20px", + width: "20px" + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "TextBlock", + text: "Workday", + size: "Small", + color: "Accent", + weight: "Bolder" + } + ], + verticalContentAlignment: "Center", + spacing: "Small" + } + ] + } + ], + verticalContentAlignment: "Center" + } + ] + } + ], + actions: [] + } + output: + binding: + actionSubmitId: Topic.actionSubmitId + endDate: Topic.endDate + hoursPerDay: Topic.hoursPerDay + startDate: Topic.startDate + timeOffType: Topic.timeOffType + + outputType: + properties: + actionSubmitId: String + endDate: String + hoursPerDay: String + startDate: String + timeOffType: String + + - kind: ConditionGroup + id: handle_cancel + conditions: + - id: cancel_pressed + condition: =Topic.actionSubmitId = "Cancel" + actions: + - kind: SendActivity + id: cancel_msg + activity: Your request has been cancelled. Is there anything else you need help with? + + - kind: CancelAllDialogs + id: cancel_all + + # Handle "I don't see my leave type listed" selection + - kind: ConditionGroup + id: handle_not_listed + conditions: + - id: type_not_listed + condition: =Topic.timeOffType = "NOT_LISTED" + actions: + - kind: SetVariable + id: set_not_listed_message + variable: Topic.notListedMessage + value: ="Since your leave type isn't listed here, you can book directly in [Workday](" & Topic.WorkdayUrl & "), or let me know if you want to change your type." + + - kind: SendActivity + id: not_listed_msg + activity: "{Topic.notListedMessage}" + + - kind: CancelAllDialogs + id: end_not_listed + + # Validate end date >= start date + - kind: ConditionGroup + id: validate_dates + conditions: + - id: end_before_start + condition: =DateValue(Topic.endDate) < DateValue(Topic.startDate) + actions: + - kind: SendActivity + id: date_error_msg + activity: The end date can't be earlier than the start date. Use this new request form to choose other dates. + + - kind: GotoAction + id: goto_collect_time_off + actionId: collect_time_off + + # ======================================================== + # V3: Build list of dates and pass to Plugin + # ======================================================== + + # Initialize date list (empty string) + - kind: SetVariable + id: init_date_list + variable: Topic.dateList + value: ="" + + # Initialize loop counter + - kind: SetVariable + id: init_iterator + variable: Topic.newIterator + value: =0 + + # Set up loop variable + - kind: SetVariable + id: set_iterator + variable: Topic.iterator + value: =Topic.newIterator + + # Calculate current date for this iteration + - kind: SetVariable + id: calc_current_date + variable: Topic.currentDate + value: =Text(DateAdd(DateValue(Topic.startDate), Topic.iterator, TimeUnit.Days), "yyyy-MM-dd") + + # Loop to build comma-separated date list + - kind: ConditionGroup + id: build_date_list_loop + conditions: + - id: should_continue_building + condition: =DateValue(Topic.currentDate) <= DateValue(Topic.endDate) + actions: + # Append date to list (with comma separator if not first) + - kind: SetVariable + id: append_date + variable: Topic.dateList + value: =If(Topic.dateList = "", Topic.currentDate, Topic.dateList & "," & Topic.currentDate) + + # Increment counter + - kind: SetVariable + id: increment_iterator + variable: Topic.newIterator + value: =Topic.newIterator + 1 + + # Continue loop + - kind: GotoAction + id: goto_build_loop + actionId: set_iterator + + # Total days is the count from loop + - kind: SetVariable + id: calc_total_days + variable: Topic.totalDays + value: =Topic.newIterator + + # Make single API call - Plugin will build XML entries from date list + - kind: BeginDialog + id: execute_workday_multiday + displayName: Submit Multi-Day Time Off Request + input: + binding: + parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{timeoff_Dates}"",""value"":""" & Topic.dateList & """},{""key"":""{timeoff_Time_Off_Type}"",""value"":""" & Topic.timeOffType & """},{""key"":""{timeoff_Hours_Per_Day}"",""value"":""" & Topic.hoursPerDay & """}]}" + scenarioName: msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay + + dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution + output: + binding: + errorResponse: Topic.errorResponse + isSuccess: Topic.isSuccess + workdayResponse: Topic.workdayResponse + + # Parse error message for display + - kind: SetVariable + id: init_last_error + variable: Topic.lastErrorMessage + value: =If(Topic.isSuccess = true, "", Topic.errorResponse) + + - kind: SetVariable + id: parse_error_message + variable: Topic.friendlyErrorMessage + value: =IfError(Text(ParseJSON(Topic.lastErrorMessage).error.message), IfError(Text(ParseJSON(Topic.lastErrorMessage).message), Topic.lastErrorMessage)) + + - kind: ConditionGroup + id: report_result + conditions: + - id: request_failed + condition: =Topic.isSuccess = false + actions: + # Use AI to generate a friendly, conversational error message + - kind: AnswerQuestionWithAI + id: generate_friendly_error + autoSend: false + variable: Topic.aiGeneratedError + userInput: =Topic.friendlyErrorMessage + additionalInstructions: Reframe the following Workday error message in a friendly way for an employee. Keep it to ONE short sentence describing what went wrong. Do NOT include suggestions or next steps. Do NOT apologize. Do NOT use technical jargon. Example output format - "The dates you selected conflict with an existing request." + + # Set final error message with fallback + - kind: SetVariable + id: set_final_error_message + variable: Topic.finalErrorMessage + value: =If(IsBlank(Topic.aiGeneratedError), "The dates you selected aren't available.", Topic.aiGeneratedError) + + - kind: SendActivity + id: friendly_error_message + activity: "{Topic.finalErrorMessage}" + + # Ask user if they want to try again + - kind: Question + id: ask_retry + interruptionPolicy: + allowInterruption: false + variable: Topic.retryChoice + prompt: Would you like to try with different dates? + entity: BooleanPrebuiltEntity + + - kind: ConditionGroup + id: handle_retry_choice + conditions: + - id: user_wants_retry + condition: =Topic.retryChoice = true + actions: + - kind: GotoAction + id: goto_form_on_retry + actionId: collect_time_off + elseActions: + - kind: SendActivity + id: end_on_no_retry + activity: No problem! Let me know if you need anything else. + + - kind: CancelAllDialogs + id: cancel_on_no_retry + + elseActions: + - kind: SendActivity + id: success_card + activity: + attachments: + - kind: AdaptiveCardTemplate + cardContent: |- + ={ + type: "AdaptiveCard", + '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", + version: "1.5", + body: [ + { + type: "TextBlock", + text: "Request submitted", + weight: "Bolder", + size: "Medium", + wrap: true + }, + { + type: "TextBlock", + text: "Your time off request has been successfully submitted and is now pending approval.", + wrap: true, + spacing: "Small" + }, + { + type: "Table", + spacing: "Medium", + showGridLines: false, + firstRowAsHeader: false, + columns: [ + { width: 1 }, + { width: 2 } + ], + rows: [ + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Type of time off", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Switch(Topic.timeOffType, "Vacation_Hours", "Vacation", "Floating_Holiday_Hours", "Floating holiday", "Sick_Hours", "Sick leave", Topic.timeOffType), wrap: true }] + } + ] + }, + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Time off date range", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Text(DateValue(Topic.startDate), "mmmm d, yyyy") & " to " & Text(DateValue(Topic.endDate), "mmmm d, yyyy") & " (" & Text(Topic.totalDays) & " days)", wrap: true }] + } + ] + }, + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Total hours", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Text(Topic.totalDays * Value(Topic.hoursPerDay)) & " hours (" & Text(Topic.totalDays) & " x " & Topic.hoursPerDay & "-hour days)", wrap: true }] + } + ] + } + ] + }, + { + type: "Container", + horizontalAlignment: "Right", + items: [ + { + type: "ColumnSet", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "Image", + url: Topic.WorkdayIconUrl, + size: "Small", + height: "20px", + width: "20px" + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "TextBlock", + text: "Workday", + size: "Small", + color: "Accent", + weight: "Bolder" + } + ], + verticalContentAlignment: "Center", + spacing: "Small" + } + ] + } + ] + } + ], + actions: [] + } + + - kind: SendActivity + id: follow_up_msg + activity: Can I help you submit another request? + + - kind: CancelAllDialogs + id: end_dialogs + +inputType: + properties: + EmployeeName: + displayName: EmployeeName + description: The name of the employee to fetch data for. + type: String + InputStartDate: + displayName: InputStartDate + description: The start date of the time off request. + type: DateTime + InputEndDate: + displayName: InputEndDate + description: The end date of the time off request. + type: DateTime + InputHoursPerDay: + displayName: InputHoursPerDay + description: The number of hours per day for the time off request. + type: Number + InputTimeOffType: + displayName: InputTimeOffType + description: The type of time off being requested. + type: String + +outputType: {} \ No newline at end of file diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml new file mode 100644 index 00000000..71949ee4 --- /dev/null +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml @@ -0,0 +1,53 @@ + + + + + User + + msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay + Absence_Management + v42.0 + + + + + //*[local-name()='Time_Off_Event_Reference']/*[local-name()='ID' and @*[local-name()='type']='WID'] + Event_WID + + + //*[local-name()='Time_Off_Entry_Reference'] + Entry_References + + + + + + + + + + false + true + true + + {timeoff_Comment} + + + + + {Employee_ID} + + + + {timeoff_Date} + {timeoff_Hours_Per_Day} + + {timeoff_Time_Off_Type} + + {timeoff_Comment} + + + + + + \ No newline at end of file From 6e12cf684925e66f3341ee934188cff5bbd4da22 Mon Sep 17 00:00:00 2001 From: Vedika Gupta Date: Wed, 4 Feb 2026 09:45:29 +0530 Subject: [PATCH 4/7] v2 instructions --- .../WorkdayGetUserProfile/topic.yaml | 31 ++++++++++--------- 1 file changed, 16 insertions(+), 15 deletions(-) diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml index 8b1c78c3..1aa7de9d 100644 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayGetUserProfile/topic.yaml @@ -1,22 +1,23 @@ kind: AdaptiveDialog modelDescription: |- - Respond to profile requests from [Workday](https://www.workday.com), an HR platform your company uses. + For FULL PROFILE: + Heading - "Your employee profile" + Introduction - "Sure, here's what I found on your employee profile in [Workday](https://www.workday.com), your company's HR platform." + Sections: Personal information, Hiring details, Role and work, Team and organization. - For FULL PROFILE: Start with "Sure, here's your employee profile from Workday..." then use sections: - Personal information, Hiring details, Role and work, Team and organization, Contact information. + For TENURE/POSITION: + Heading - "Time in your position" + Introduction - "Here's how long you've been in your position according to [Workday](https://www.workday.com), an HR platform your company uses." + Verbatim line after heading: "You've been in your current role as a [Position] for [LengthOfService]." + Sections: About your position, Previous history with the organization (only if available). - For TENURE/POSITION questions: Start with "Here's how long you've been in your position according to Workday..." - Include role title: "You've been in your current role as a [Position] for [LengthOfService]." - Add "About your position" section: Department, Location. - - Rules: - - Always include Workday attribution - - Use section headers, bullet lists within sections - - Home address as single line, NOT sub-bullets - - Exclude DOB/Gender unless asked - - End with: "Let me know if you want more details or to update your profile." - - Invalid: other people's data. + Common Rules (Strict): + - Large heading & section line after it, + - Use sections mentioned as headers & bullets within + - ADDRESS IN ONE ROW + - Introduction - BEFORE heading + - Include relevant data, exclude gender + - Only end with "Let me know if you want more details about a specific section or want to update something." add section line befoe it beginDialog: kind: OnRecognizedIntent From 970df7c2799ba4d46a6367a6d45082b042955e6c Mon Sep 17 00:00:00 2001 From: Vedika Gupta Date: Wed, 4 Feb 2026 14:41:31 +0530 Subject: [PATCH 5/7] v2 for direct reportees time --- .../topic.yaml | 24 ++++++++++++------- 1 file changed, 16 insertions(+), 8 deletions(-) diff --git a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml index 90453187..e20f1f34 100644 --- a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml @@ -1,13 +1,21 @@ kind: AdaptiveDialog modelDescription: |- - Respond to team time-in-position requests from [Workday](https://www.workday.com), an HR platform your company uses. - - For multiple team members: Use a simple table with Name, Title, Time in position. - For one team member: Use their name as header with bullet list of details. - - Keep responses concise. Do NOT show Status or Hire Date unless asked. - End with: "Let me know if you have questions about your team." - + You will respond to requests about time in position for direct reports of the user making the request. + + For info on single direct report's time in position: + Heading - "[Report]'s time in position" + Verbatim line after heading: "[Report] has been in their current role as a [Position] for [LengthOfService]." + Sections as headers and bullets within: More about [Direct Report]'s position, Previous history with the organization (only if available). + + For info on all direct reports time in position: Display in table + Heading - "Team roles and time in position" + Table columns: Name, Title, Time in Position + + Common Rules (STRICT): + - Large heading & section line after it, + - Introduction (Relevant)- "Here's how long [Report] has been in their position according to [Workday](https://www.workday.com), an HR platform your company uses." + - Introduction - ONLY BEFORE heading + - Only end with "Let me know if you want more details about your team or a specific section." add section line befoe it Invalid: non-direct reports. beginDialog: kind: OnRecognizedIntent From 75210b34b1b1209bbfd038da8e88f0d781b43db6 Mon Sep 17 00:00:00 2001 From: Vedika Gupta Date: Wed, 4 Feb 2026 16:40:16 +0530 Subject: [PATCH 6/7] nit changes --- .../WorkdayGetManagerReporteesTimeInPosition/topic.yaml | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml index e20f1f34..25fb49c3 100644 --- a/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/ManagerScenarios/WorkdayGetManagerReporteesTimeInPosition/topic.yaml @@ -11,11 +11,10 @@ modelDescription: |- Heading - "Team roles and time in position" Table columns: Name, Title, Time in Position - Common Rules (STRICT): - - Large heading & section line after it, - - Introduction (Relevant)- "Here's how long [Report] has been in their position according to [Workday](https://www.workday.com), an HR platform your company uses." - - Introduction - ONLY BEFORE heading - - Only end with "Let me know if you want more details about your team or a specific section." add section line befoe it + Common Rules (STRICT!): + - ALWAYS KEEP LARGE HEADING & ADD SECTION LINE AFTER IT, + - Introduction (Relevant) ONLY BEFORE heading - "Here's how long [Report] has been in their position according to [Workday](https://www.workday.com), an HR platform your company uses." + - Only end with "Let me know if you want more details about your team or a specific section." ADD SECTION LINE BEFORE IT. Invalid: non-direct reports. beginDialog: kind: OnRecognizedIntent From b134abe85bf2122df8d52735a4865501c5054895 Mon Sep 17 00:00:00 2001 From: Ankur Rana Date: Wed, 4 Feb 2026 16:58:20 +0530 Subject: [PATCH 7/7] Add README and multi-day support for WorkdayEmployeeRequestTimeOff topic --- .../RequestTimeOff_Topic.yaml | 741 ------------------ .../WorkdayEmployeeRequestTimeOff/README.md | 177 +++++ ...ayAbsenceEnterTimeOff_EnterTimeOffInfo.xml | 48 -- ..._HRWorkdayAbsenceEnterTimeOff_MultiDay.xml | 2 + .../WorkdayEmployeeRequestTimeOff/topic.yaml | 721 +++++++++++++++-- 5 files changed, 819 insertions(+), 870 deletions(-) delete mode 100644 EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml create mode 100644 EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/README.md delete mode 100644 EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo.xml rename EmployeeSelfServiceAgent/Workday/EmployeeScenarios/{WorkdayEmployeeRequestTimeOff-V2 => WorkdayEmployeeRequestTimeOff}/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml (99%) diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml deleted file mode 100644 index 7b5849b9..00000000 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/RequestTimeOff_Topic.yaml +++ /dev/null @@ -1,741 +0,0 @@ -kind: AdaptiveDialog -inputs: - - kind: AutomaticTaskInput - propertyName: EmployeeName - description: The name of the employee to fetch data for. - entity: PersonNamePrebuiltEntity - shouldPromptUser: false - - - kind: AutomaticTaskInput - propertyName: InputStartDate - description: The start date of the time off request. - entity: DateTimePrebuiltEntity - shouldPromptUser: false - - - kind: AutomaticTaskInput - propertyName: InputEndDate - description: The end date of the time off request. - entity: DateTimePrebuiltEntity - shouldPromptUser: false - - - kind: AutomaticTaskInput - propertyName: InputHoursPerDay - description: The number of hours per day for the time off request. - entity: NumberPrebuiltEntity - shouldPromptUser: false - - - kind: AutomaticTaskInput - propertyName: InputTimeOffType - description: Extract the type of leave mentioned such as vacation, sick, sick leave, floating holiday, floater, PTO, annual leave, medical, or illness. Extract keywords like 'sick', 'vacation', 'floating', 'PTO', 'annual'. - entity: StringPrebuiltEntity - shouldPromptUser: false - -modelDescription: |- - You will respond only to requests related to requesting time off for the user making the request. - All time off requests are submitted to Workday (Absence_Management) and pertain exclusively to the requesting user. - This topic may prompt the user for input (time off type, start date, end date, and reason) if they are requesting time off for themselves. - This data exclusively belongs to the user making the request. Do not respond to questions about other people's data. - - Example invalid requests: - "Request time off for my manager" - "Request time off for my sister" - "Request time off for [EmployeeName]" - - Example valid requests: - "Request time off" - "I need to submit time off" - "Please request vacation from January 5th to January 10th" - "Request sick leave for next week" - "I need time off from 2025-09-15 to 2025-09-20 for a family event" -beginDialog: - kind: OnRecognizedIntent - id: main - intent: {} - actions: - - kind: SetVariable - id: set_workday_url - variable: Topic.WorkdayUrl - value: https://impl.workday.com/microsoft_dpt6/d/home.htmld - - - kind: SetVariable - id: set_workday_icon_url - variable: Topic.WorkdayIconUrl - value: https://conn-afd-prod-endpoint-bmc9bqahasf3grgk.b01.azurefd.net/v1.0.1774/1.0.1774.4397/workdaysoap/icon.png - - # ================================================================ - # DATE CONFIGURATION - # Set the effective date for balance lookup. Default is Today(). - # Customers can change this to a specific date if needed. - # ================================================================ - - kind: SetVariable - id: set_as_of_effective_date - variable: Topic.AsOfEffectiveDate - value: =Today() - - # ================================================================ - # LEAVE TYPE CONFIGURATION - # Edit keywords below to customize how user phrases map to leave types. - # Keywords are comma-separated and case-insensitive. - # LeaveTypeValue must match the dropdown choice values. - # ================================================================ - - kind: SetVariable - id: set_leave_type_config - variable: Topic.LeaveTypeConfig - value: |- - =Table( - {Keywords: "vacation,annual,pto,holiday pay", LeaveTypeValue: "Vacation_Hours"}, - {Keywords: "floating,floater,float day", LeaveTypeValue: "Floating_Holiday_Hours"}, - {Keywords: "sick,illness,medical,unwell,not feeling well", LeaveTypeValue: "Sick_Hours"} - ) - - # Map extracted time off type to dropdown value using config table - - kind: SetVariable - id: set_mapped_time_off_type - variable: Topic.MappedTimeOffType - value: |- - =If( - IsBlank(Topic.InputTimeOffType), - "", - Coalesce( - First( - Filter( - Topic.LeaveTypeConfig, - !IsEmpty(Filter(Split(Keywords, ","), Trim(Value) in Lower(Topic.InputTimeOffType))) - ) - ).LeaveTypeValue, - "" - ) - ) - - - kind: BeginDialog - id: get_leave_balance - displayName: Get Leave Balance - input: - binding: - parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{As_Of_Effective_Date}"",""value"":""" & Text(Topic.AsOfEffectiveDate, "yyyy-MM-dd") & """}]}" - scenarioName: msdyn_HRWorkdayHCMEmployeeGetVacationBalance - - dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution - output: - binding: - errorResponse: Topic.balanceErrorResponse - isSuccess: Topic.balanceIsSuccess - workdayResponse: Topic.balanceResponse - - - kind: ParseValue - id: parse_balance - variable: Topic.parsedBalance - valueType: - kind: Record - properties: - PlanDescriptor: - type: - kind: Table - properties: - Value: String - - PlanID: - type: - kind: Table - properties: - Value: String - - RemainingBalance: - type: - kind: Table - properties: - Value: String - - UnitOfTime: - type: - kind: Table - properties: - Value: String - - value: =Topic.balanceResponse - - # ================================================================ - # PLAN BALANCE CONFIGURATION - # Maps Plan IDs (from balance API) to display names. - # Update PlanID values to match your Workday configuration. - # Only plans listed here AND returned by the API will be displayed. - # ================================================================ - - kind: SetVariable - id: set_plan_config - variable: Topic.PlanConfig - value: |- - =Table( - {PlanID: "PTO_USA", DisplayName: "Paid time off"}, - {PlanID: "FH_USA", DisplayName: "Floating holiday"}, - {PlanID: "ABSENCE_PLAN-6-159", DisplayName: "Sick leave"}, - {PlanID: "ABSENCE_PLAN-6-158", DisplayName: "Vacation"} - ) - - # Create merged balance table for easy lookup by PlanID - - kind: SetVariable - id: set_balance_table - variable: Topic.BalanceTable - value: |- - =ForAll( - Sequence(CountRows(Topic.parsedBalance.PlanID)), - { - PlanID: Index(Topic.parsedBalance.PlanID, Value).Value, - Balance: Index(Topic.parsedBalance.RemainingBalance, Value).Value - } - ) - - # Build display data: only include plans that exist in BOTH config AND API response - - kind: SetVariable - id: set_display_balances - variable: Topic.DisplayBalances - value: |- - =ForAll( - Filter( - Topic.PlanConfig, - !IsBlank(LookUp(Topic.BalanceTable, PlanID = ThisRecord.PlanID).Balance) - ) As plan, - { - PlanID: plan.PlanID, - DisplayName: plan.DisplayName, - Balance: LookUp(Topic.BalanceTable, PlanID = plan.PlanID).Balance - } - ) - - - kind: SetVariable - id: build_intro_message - variable: Topic.introMessage - value: ="Sure, I'll help you submit a time off request. Here's a form where you can choose the type of leave and dates you want off. Don't see the type of leave you want? [Book directly in Workday](" & Topic.WorkdayUrl & ")" - - - kind: SendActivity - id: intro_message - activity: "{Topic.introMessage}" - - - kind: ConditionGroup - id: need_inputs - conditions: - - id: always_true - condition: true - actions: - - kind: AdaptiveCardPrompt - id: collect_time_off - displayName: Ask time off type, start date, end date and reason - card: |- - ={ - type: "AdaptiveCard", - '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", - version: "1.5", - body: [ - { - type: "TextBlock", - text: "Book your time off", - weight: "Bolder", - size: "Medium", - wrap: true, - color: "Default" - }, - { - type: "Container", - style: "emphasis", - bleed: true, - items: [ - { - type: "TextBlock", - text: "Available balance in hours as on " & Text(Topic.AsOfEffectiveDate, "mmmm d, yyyy"), - size: "Small", - weight: "Bolder", - wrap: true - }, - { - type: "ColumnSet", - columns: ForAll( - Topic.DisplayBalances, - { - type: "Column", - width: "stretch", - items: [ - { type: "TextBlock", text: DisplayName, size: "Small", wrap: true }, - { type: "TextBlock", text: Balance, weight: "Bolder", spacing: "Small" } - ] - } - ) - } - ] - }, - { - type: "TextBlock", - text: "Fields marked with * are required.", - wrap: true, - spacing: "Medium", - size: "Small" - }, - { - type: "ColumnSet", - columns: [ - { - type: "Column", - width: "stretch", - items: [ - { - type: "Input.ChoiceSet", - id: "timeOffType", - label: "Type of time off", - style: "compact", - value: Topic.MappedTimeOffType, - isRequired: true, - errorMessage: "Please select a type.", - choices: [ - { title: "Vacation", value: "Vacation_Hours" }, - { title: "Floating holiday", value: "Floating_Holiday_Hours" }, - { title: "Sick leave", value: "Sick_Hours" }, - { title: "I don't see my leave type listed", value: "NOT_LISTED" } - ] - } - ] - } - ] - }, - { - type: "Input.Date", - id: "startDate", - label: "Start date", - value: If(IsBlank(Topic.InputStartDate), "", Text(Topic.InputStartDate, "yyyy-MM-dd")), - isRequired: true, - errorMessage: "Please select a start date." - }, - { - type: "Input.Date", - id: "endDate", - label: "End date", - value: If(IsBlank(Topic.InputEndDate), "", Text(Topic.InputEndDate, "yyyy-MM-dd")), - isRequired: true, - errorMessage: "Please select an end date." - }, - { - type: "Input.Number", - id: "hoursPerDay", - label: "Hours per day", - min: 1, - max: 24, - value: If(IsBlank(Topic.InputHoursPerDay), 8, Value(Topic.InputHoursPerDay)), - isRequired: true, - errorMessage: "Please enter hours per day." - }, - { - type: "ColumnSet", - spacing: "Medium", - columns: [ - { - type: "Column", - width: "auto", - items: [ - { - type: "ActionSet", - actions: [ - { type: "Action.Submit", title: "Submit" } - ] - } - ], - verticalContentAlignment: "Center" - }, - { - type: "Column", - width: "stretch", - items: [], - verticalContentAlignment: "Center" - }, - { - type: "Column", - width: "auto", - items: [ - { - type: "ColumnSet", - spacing: "None", - columns: [ - { - type: "Column", - width: "auto", - items: [ - { - type: "Image", - url: Topic.WorkdayIconUrl, - size: "Small", - height: "20px", - width: "20px" - } - ], - verticalContentAlignment: "Center" - }, - { - type: "Column", - width: "auto", - items: [ - { - type: "TextBlock", - text: "Workday", - size: "Small", - color: "Accent", - weight: "Bolder" - } - ], - verticalContentAlignment: "Center", - spacing: "Small" - } - ] - } - ], - verticalContentAlignment: "Center" - } - ] - } - ], - actions: [] - } - output: - binding: - actionSubmitId: Topic.actionSubmitId - endDate: Topic.endDate - hoursPerDay: Topic.hoursPerDay - startDate: Topic.startDate - timeOffType: Topic.timeOffType - - outputType: - properties: - actionSubmitId: String - endDate: String - hoursPerDay: String - startDate: String - timeOffType: String - - - kind: ConditionGroup - id: handle_cancel - conditions: - - id: cancel_pressed - condition: =Topic.actionSubmitId = "Cancel" - actions: - - kind: SendActivity - id: cancel_msg - activity: Your request has been cancelled. Is there anything else you need help with? - - - kind: CancelAllDialogs - id: cancel_all - - # Handle "I don't see my leave type listed" selection - - kind: ConditionGroup - id: handle_not_listed - conditions: - - id: type_not_listed - condition: =Topic.timeOffType = "NOT_LISTED" - actions: - - kind: SetVariable - id: set_not_listed_message - variable: Topic.notListedMessage - value: ="Since your leave type isn't listed here, you can book directly in [Workday](" & Topic.WorkdayUrl & "), or let me know if you want to change your type." - - - kind: SendActivity - id: not_listed_msg - activity: "{Topic.notListedMessage}" - - - kind: CancelAllDialogs - id: end_not_listed - - # Validate end date >= start date - - kind: ConditionGroup - id: validate_dates - conditions: - - id: end_before_start - condition: =DateValue(Topic.endDate) < DateValue(Topic.startDate) - actions: - - kind: SendActivity - id: date_error_msg - activity: The end date can't be earlier than the start date. Use this new request form to choose other dates. - - - kind: GotoAction - id: goto_collect_time_off - actionId: collect_time_off - - # ======================================================== - # V3: Build list of dates and pass to Plugin - # ======================================================== - - # Initialize date list (empty string) - - kind: SetVariable - id: init_date_list - variable: Topic.dateList - value: ="" - - # Initialize loop counter - - kind: SetVariable - id: init_iterator - variable: Topic.newIterator - value: =0 - - # Set up loop variable - - kind: SetVariable - id: set_iterator - variable: Topic.iterator - value: =Topic.newIterator - - # Calculate current date for this iteration - - kind: SetVariable - id: calc_current_date - variable: Topic.currentDate - value: =Text(DateAdd(DateValue(Topic.startDate), Topic.iterator, TimeUnit.Days), "yyyy-MM-dd") - - # Loop to build comma-separated date list - - kind: ConditionGroup - id: build_date_list_loop - conditions: - - id: should_continue_building - condition: =DateValue(Topic.currentDate) <= DateValue(Topic.endDate) - actions: - # Append date to list (with comma separator if not first) - - kind: SetVariable - id: append_date - variable: Topic.dateList - value: =If(Topic.dateList = "", Topic.currentDate, Topic.dateList & "," & Topic.currentDate) - - # Increment counter - - kind: SetVariable - id: increment_iterator - variable: Topic.newIterator - value: =Topic.newIterator + 1 - - # Continue loop - - kind: GotoAction - id: goto_build_loop - actionId: set_iterator - - # Total days is the count from loop - - kind: SetVariable - id: calc_total_days - variable: Topic.totalDays - value: =Topic.newIterator - - # Make single API call - Plugin will build XML entries from date list - - kind: BeginDialog - id: execute_workday_multiday - displayName: Submit Multi-Day Time Off Request - input: - binding: - parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{timeoff_Dates}"",""value"":""" & Topic.dateList & """},{""key"":""{timeoff_Time_Off_Type}"",""value"":""" & Topic.timeOffType & """},{""key"":""{timeoff_Hours_Per_Day}"",""value"":""" & Topic.hoursPerDay & """}]}" - scenarioName: msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay - - dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution - output: - binding: - errorResponse: Topic.errorResponse - isSuccess: Topic.isSuccess - workdayResponse: Topic.workdayResponse - - # Parse error message for display - - kind: SetVariable - id: init_last_error - variable: Topic.lastErrorMessage - value: =If(Topic.isSuccess = true, "", Topic.errorResponse) - - - kind: SetVariable - id: parse_error_message - variable: Topic.friendlyErrorMessage - value: =IfError(Text(ParseJSON(Topic.lastErrorMessage).error.message), IfError(Text(ParseJSON(Topic.lastErrorMessage).message), Topic.lastErrorMessage)) - - - kind: ConditionGroup - id: report_result - conditions: - - id: request_failed - condition: =Topic.isSuccess = false - actions: - # Use AI to generate a friendly, conversational error message - - kind: AnswerQuestionWithAI - id: generate_friendly_error - autoSend: false - variable: Topic.aiGeneratedError - userInput: =Topic.friendlyErrorMessage - additionalInstructions: Reframe the following Workday error message in a friendly way for an employee. Keep it to ONE short sentence describing what went wrong. Do NOT include suggestions or next steps. Do NOT apologize. Do NOT use technical jargon. Example output format - "The dates you selected conflict with an existing request." - - # Set final error message with fallback - - kind: SetVariable - id: set_final_error_message - variable: Topic.finalErrorMessage - value: =If(IsBlank(Topic.aiGeneratedError), "The dates you selected aren't available.", Topic.aiGeneratedError) - - - kind: SendActivity - id: friendly_error_message - activity: "{Topic.finalErrorMessage}" - - # Ask user if they want to try again - - kind: Question - id: ask_retry - interruptionPolicy: - allowInterruption: false - variable: Topic.retryChoice - prompt: Would you like to try with different dates? - entity: BooleanPrebuiltEntity - - - kind: ConditionGroup - id: handle_retry_choice - conditions: - - id: user_wants_retry - condition: =Topic.retryChoice = true - actions: - - kind: GotoAction - id: goto_form_on_retry - actionId: collect_time_off - elseActions: - - kind: SendActivity - id: end_on_no_retry - activity: No problem! Let me know if you need anything else. - - - kind: CancelAllDialogs - id: cancel_on_no_retry - - elseActions: - - kind: SendActivity - id: success_card - activity: - attachments: - - kind: AdaptiveCardTemplate - cardContent: |- - ={ - type: "AdaptiveCard", - '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", - version: "1.5", - body: [ - { - type: "TextBlock", - text: "Request submitted", - weight: "Bolder", - size: "Medium", - wrap: true - }, - { - type: "TextBlock", - text: "Your time off request has been successfully submitted and is now pending approval.", - wrap: true, - spacing: "Small" - }, - { - type: "Table", - spacing: "Medium", - showGridLines: false, - firstRowAsHeader: false, - columns: [ - { width: 1 }, - { width: 2 } - ], - rows: [ - { - type: "TableRow", - cells: [ - { - type: "TableCell", - items: [{ type: "TextBlock", text: "Type of time off", weight: "Bolder" }] - }, - { - type: "TableCell", - items: [{ type: "TextBlock", text: Switch(Topic.timeOffType, "Vacation_Hours", "Vacation", "Floating_Holiday_Hours", "Floating holiday", "Sick_Hours", "Sick leave", Topic.timeOffType), wrap: true }] - } - ] - }, - { - type: "TableRow", - cells: [ - { - type: "TableCell", - items: [{ type: "TextBlock", text: "Time off date range", weight: "Bolder" }] - }, - { - type: "TableCell", - items: [{ type: "TextBlock", text: Text(DateValue(Topic.startDate), "mmmm d, yyyy") & " to " & Text(DateValue(Topic.endDate), "mmmm d, yyyy") & " (" & Text(Topic.totalDays) & " days)", wrap: true }] - } - ] - }, - { - type: "TableRow", - cells: [ - { - type: "TableCell", - items: [{ type: "TextBlock", text: "Total hours", weight: "Bolder" }] - }, - { - type: "TableCell", - items: [{ type: "TextBlock", text: Text(Topic.totalDays * Value(Topic.hoursPerDay)) & " hours (" & Text(Topic.totalDays) & " x " & Topic.hoursPerDay & "-hour days)", wrap: true }] - } - ] - } - ] - }, - { - type: "Container", - horizontalAlignment: "Right", - items: [ - { - type: "ColumnSet", - columns: [ - { - type: "Column", - width: "auto", - items: [ - { - type: "Image", - url: Topic.WorkdayIconUrl, - size: "Small", - height: "20px", - width: "20px" - } - ], - verticalContentAlignment: "Center" - }, - { - type: "Column", - width: "auto", - items: [ - { - type: "TextBlock", - text: "Workday", - size: "Small", - color: "Accent", - weight: "Bolder" - } - ], - verticalContentAlignment: "Center", - spacing: "Small" - } - ] - } - ] - } - ], - actions: [] - } - - - kind: SendActivity - id: follow_up_msg - activity: Can I help you submit another request? - - - kind: CancelAllDialogs - id: end_dialogs - -inputType: - properties: - EmployeeName: - displayName: EmployeeName - description: The name of the employee to fetch data for. - type: String - InputStartDate: - displayName: InputStartDate - description: The start date of the time off request. - type: DateTime - InputEndDate: - displayName: InputEndDate - description: The end date of the time off request. - type: DateTime - InputHoursPerDay: - displayName: InputHoursPerDay - description: The number of hours per day for the time off request. - type: Number - InputTimeOffType: - displayName: InputTimeOffType - description: The type of time off being requested. - type: String - -outputType: {} \ No newline at end of file diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/README.md b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/README.md new file mode 100644 index 00000000..4e0183f3 --- /dev/null +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/README.md @@ -0,0 +1,177 @@ +# Request Time Off + +This scenario allows employees to submit time off requests to Workday, supporting single-day or multi-day date ranges with automatic balance display. + +## Overview + +The `WorkdayEmployeeRequestTimeOff` topic allows employees to: +1. **View** their current leave balances (vacation, sick, floating holiday) +2. **Submit** time off requests for a date range +3. **Choose** the type of leave and hours per day + +## Flow + +``` +┌─────────────────────────────────────┐ +│ User triggers topic │ +│ "Request time off" │ +└──────────────┬──────────────────────┘ + │ + ▼ +┌─────────────────────────────────────┐ +│ Fetch current leave balances │ +│ (msdyn_HRWorkdayHCMEmployee │ +│ GetVacationBalance) │ +└──────────────┬──────────────────────┘ + │ + ▼ +┌─────────────────────────────────────┐ +│ Display Adaptive Card Form │ +│ - Shows available balances │ +│ - Time off type dropdown │ +│ - Start date / End date │ +│ - Hours per day │ +└──────────────┬──────────────────────┘ + │ + ▼ + ┌──────┴──────┐ + │ User │ + │ Action? │ + └──────┬──────┘ + │ + ┌───────┼───────┐ + │ │ │ + ▼ ▼ ▼ +┌──────────┐ ┌──────────┐ ┌──────────────┐ +│ Cancel │ │ Type Not │ │ Submit │ +│ │ │ Listed │ │ │ +└────┬─────┘ └────┬─────┘ └──────┬───────┘ + │ │ │ + ▼ ▼ ▼ +┌──────────┐ ┌──────────┐ ┌──────────────┐ +│ End │ │ Redirect │ │ Validate │ +│ Dialog │ │ to │ │ Dates │ +└──────────┘ │ Workday │ └──────┬───────┘ + └──────────┘ │ + ▼ + ┌────────────────────────┐ + │ Build date list │ + │ (loop: start → end) │ + └────────────┬───────────┘ + │ + ▼ + ┌────────────────────────┐ + │ Submit to Workday │ + │ (msdyn_HRWorkday │ + │ AbsenceEnterTimeOff │ + │ _MultiDay) │ + └────────────┬───────────┘ + │ + ┌───────┴───────┐ + │ │ + ▼ ▼ + ┌──────────┐ ┌──────────────┐ + │ Success │ │ Error │ + │ Card │ │ (AI-friendly │ + │ │ │ message) │ + └──────────┘ └──────────────┘ +``` + +## API Scenarios Used + +| Scenario | API | Purpose | +|----------|-----|---------| +| `msdyn_HRWorkdayHCMEmployeeGetVacationBalance` | Human_Resources | Fetch current leave balances | +| `msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay` | Absence_Management v42.0 | Submit time off request | + +## Features + +### Intelligent Leave Type Mapping +User input is automatically mapped to leave types using configurable keywords: + +| User Says | Maps To | +|-----------|---------| +| "vacation", "annual", "pto", "holiday pay" | Vacation_Hours | +| "floating", "floater", "float day" | Floating_Holiday_Hours | +| "sick", "illness", "medical", "unwell" | Sick_Hours | + +## Form Fields + +| Field | Type | Required | Default | Description | +|-------|------|----------|---------|-------------| +| Type of time off | Dropdown | Yes | Auto-mapped | Vacation, Floating holiday, Sick leave | +| Start date | Date Input | Yes | From user query | First day of time off | +| End date | Date Input | Yes | From user query | Last day of time off | +| Hours per day | Number Input | Yes | 8 | Hours to deduct per day | + +## Input Parameters + +| Parameter | Type | Description | +|-----------|------|-------------| +| `InputStartDate` | DateTime | Pre-fill start date from user query | +| `InputEndDate` | DateTime | Pre-fill end date from user query | +| `InputHoursPerDay` | Number | Pre-fill hours per day | +| `InputTimeOffType` | String | Natural language leave type (e.g., "vacation", "sick") | + +## XML Template Parameters + +| Parameter | Description | Example | +|-----------|-------------|---------| +| `{Employee_ID}` | Employee's Workday ID | `21514` | +| `{timeoff_Dates}` | Comma-separated list of dates | `2025-09-15,2025-09-16,2025-09-17` | +| `{timeoff_Time_Off_Type}` | Time off type ID | `Vacation_Hours` | +| `{timeoff_Hours_Per_Day}` | Hours per day | `8` | +| `{timeoff_Comment}` | Comment for the request | `ess generated time off request` | + +## Configuration + +### Leave Type Configuration +Edit the `LeaveTypeConfig` variable to customize keyword-to-leave-type mappings: + +```yaml +- kind: SetVariable + id: set_leave_type_config + variable: Topic.LeaveTypeConfig + value: |- + =Table( + {Keywords: "vacation,annual,pto,holiday pay", LeaveTypeValue: "Vacation_Hours"}, + {Keywords: "floating,floater,float day", LeaveTypeValue: "Floating_Holiday_Hours"}, + {Keywords: "sick,illness,medical,unwell,not feeling well", LeaveTypeValue: "Sick_Hours"} + ) +``` + +### Plan Balance Configuration +Edit the `PlanConfig` variable to map Workday Plan IDs to display names: + +```yaml +- kind: SetVariable + id: set_plan_config + variable: Topic.PlanConfig + value: |- + =Table( + {PlanID: "PTO_USA", DisplayName: "Paid time off"}, + {PlanID: "FH_USA", DisplayName: "Floating holiday"}, + {PlanID: "ABSENCE_PLAN-6-159", DisplayName: "Sick leave"}, + {PlanID: "ABSENCE_PLAN-6-158", DisplayName: "Vacation"} + ) +``` + +### Workday URL Configuration +Update the `WorkdayUrl` variable to point to your Workday tenant: + +```yaml +- kind: SetVariable + id: set_workday_url + variable: Topic.WorkdayUrl + value: https://impl.workday.com//home.htmld +``` + +## Example Triggers + +**Valid requests:** +- "Request time off" +- "I need to submit time off" +- "Please request vacation from January 5th to January 10th" +- "Request sick leave for next week" +- "I need time off from 2025-09-15 to 2025-09-20 for a family event" +- "Book 4 hours of PTO tomorrow" diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo.xml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo.xml deleted file mode 100644 index 4b8ad55e..00000000 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo.xml +++ /dev/null @@ -1,48 +0,0 @@ - - - - - User - - msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo - Absence_Management - v42.0 - - - - - //*[local-name()='Time_Off_Event_Reference']/*[local-name()='ID' and @*[local-name()='type']='WID'] - WID - - - - - - - - - - false - true - true - - {Comment} - - - - - {Employee_ID} - - - {Time_Off_Date} - {Hours} - - {Reason_ID} - - {Comment} - - - - - - \ No newline at end of file diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml similarity index 99% rename from EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml rename to EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml index 71949ee4..7c6cc0ee 100644 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff-V2/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay.xml @@ -38,6 +38,7 @@ {Employee_ID} + {timeoff_Date} {timeoff_Hours_Per_Day} @@ -46,6 +47,7 @@ {timeoff_Comment} + diff --git a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/topic.yaml b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/topic.yaml index 1921a489..67885e78 100644 --- a/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/topic.yaml +++ b/EmployeeSelfServiceAgent/Workday/EmployeeScenarios/WorkdayEmployeeRequestTimeOff/topic.yaml @@ -1,32 +1,39 @@ kind: AdaptiveDialog inputs: - kind: AutomaticTaskInput - propertyName: TimeOffDate - description: Date of time off (yyyy-mm-dd) + propertyName: EmployeeName + description: The name of the employee to fetch data for. + entity: PersonNamePrebuiltEntity + shouldPromptUser: false + + - kind: AutomaticTaskInput + propertyName: InputStartDate + description: The start date of the time off request. entity: DateTimePrebuiltEntity shouldPromptUser: false - kind: AutomaticTaskInput - propertyName: NumberOfHours - description: Number of hours for time off - entity: NumberPrebuiltEntity + propertyName: InputEndDate + description: The end date of the time off request. + entity: DateTimePrebuiltEntity shouldPromptUser: false - kind: AutomaticTaskInput - propertyName: ReasonText - description: Reason for time off (free text) + propertyName: InputHoursPerDay + description: The number of hours per day for the time off request. + entity: NumberPrebuiltEntity shouldPromptUser: false - kind: AutomaticTaskInput - propertyName: EmployeeName - description: The name of the employee to fetch data for. - entity: PersonNamePrebuiltEntity + propertyName: InputTimeOffType + description: Extract the type of leave mentioned such as vacation, sick, sick leave, floating holiday, floater, PTO, annual leave, medical, or illness. Extract keywords like 'sick', 'vacation', 'floating', 'PTO', 'annual'. + entity: StringPrebuiltEntity shouldPromptUser: false modelDescription: |- You will respond only to requests related to requesting time off for the user making the request. All time off requests are submitted to Workday (Absence_Management) and pertain exclusively to the requesting user. - This topic may prompt the user for input (date, hours, and reason) if they are requesting time off for themselves. + This topic may prompt the user for input (time off type, start date, end date, and reason) if they are requesting time off for themselves. This data exclusively belongs to the user making the request. Do not respond to questions about other people's data. Example invalid requests: @@ -37,13 +44,172 @@ modelDescription: |- Example valid requests: "Request time off" "I need to submit time off" - "Please request 8 hours time off on 2025-09-15 because of a family event" - "Request 4 hours vacation on December 1st" + "Please request vacation from January 5th to January 10th" + "Request sick leave for next week" + "I need time off from 2025-09-15 to 2025-09-20 for a family event" beginDialog: kind: OnRecognizedIntent id: main intent: {} actions: + - kind: SetVariable + id: set_workday_url + variable: Topic.WorkdayUrl + value: https://impl.workday.com//home.htmld + + - kind: SetVariable + id: set_workday_icon_url + variable: Topic.WorkdayIconUrl + value:  + + # ================================================================ + # DATE CONFIGURATION + # Set the effective date for balance lookup. Default is Today(). + # Customers can change this to a specific date if needed. + # ================================================================ + - kind: SetVariable + id: set_as_of_effective_date + variable: Topic.AsOfEffectiveDate + value: =Today() + + # ================================================================ + # LEAVE TYPE CONFIGURATION + # Edit keywords below to customize how user phrases map to leave types. + # Keywords are comma-separated and case-insensitive. + # LeaveTypeValue must match the dropdown choice values. + # ================================================================ + - kind: SetVariable + id: set_leave_type_config + variable: Topic.LeaveTypeConfig + value: |- + =Table( + {Keywords: "vacation,annual,pto,holiday pay", LeaveTypeValue: "Vacation_Hours"}, + {Keywords: "floating,floater,float day", LeaveTypeValue: "Floating_Holiday_Hours"}, + {Keywords: "sick,illness,medical,unwell,not feeling well", LeaveTypeValue: "Sick_Hours"} + ) + + # Map extracted time off type to dropdown value using config table + - kind: SetVariable + id: set_mapped_time_off_type + variable: Topic.MappedTimeOffType + value: |- + =If( + IsBlank(Topic.InputTimeOffType), + "", + Coalesce( + First( + Filter( + Topic.LeaveTypeConfig, + !IsEmpty(Filter(Split(Keywords, ","), Trim(Value) in Lower(Topic.InputTimeOffType))) + ) + ).LeaveTypeValue, + "" + ) + ) + + - kind: BeginDialog + id: get_leave_balance + displayName: Get Leave Balance + input: + binding: + parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{As_Of_Effective_Date}"",""value"":""" & Text(Topic.AsOfEffectiveDate, "yyyy-MM-dd") & """}]}" + scenarioName: msdyn_HRWorkdayHCMEmployeeGetVacationBalance + + dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution + output: + binding: + errorResponse: Topic.balanceErrorResponse + isSuccess: Topic.balanceIsSuccess + workdayResponse: Topic.balanceResponse + + - kind: ParseValue + id: parse_balance + variable: Topic.parsedBalance + valueType: + kind: Record + properties: + PlanDescriptor: + type: + kind: Table + properties: + Value: String + + PlanID: + type: + kind: Table + properties: + Value: String + + RemainingBalance: + type: + kind: Table + properties: + Value: String + + UnitOfTime: + type: + kind: Table + properties: + Value: String + + value: =Topic.balanceResponse + + # ================================================================ + # PLAN BALANCE CONFIGURATION + # Maps Plan IDs (from balance API) to display names. + # Update PlanID values to match your Workday configuration. + # Only plans listed here AND returned by the API will be displayed. + # ================================================================ + - kind: SetVariable + id: set_plan_config + variable: Topic.PlanConfig + value: |- + =Table( + {PlanID: "PTO_USA", DisplayName: "Paid time off"}, + {PlanID: "FH_USA", DisplayName: "Floating holiday"}, + {PlanID: "ABSENCE_PLAN-6-159", DisplayName: "Sick leave"}, + {PlanID: "ABSENCE_PLAN-6-158", DisplayName: "Vacation"} + ) + + # Create merged balance table for easy lookup by PlanID + - kind: SetVariable + id: set_balance_table + variable: Topic.BalanceTable + value: |- + =ForAll( + Sequence(CountRows(Topic.parsedBalance.PlanID)), + { + PlanID: Index(Topic.parsedBalance.PlanID, Value).Value, + Balance: Index(Topic.parsedBalance.RemainingBalance, Value).Value + } + ) + + # Build display data: only include plans that exist in BOTH config AND API response + - kind: SetVariable + id: set_display_balances + variable: Topic.DisplayBalances + value: |- + =ForAll( + Filter( + Topic.PlanConfig, + !IsBlank(LookUp(Topic.BalanceTable, PlanID = ThisRecord.PlanID).Balance) + ) As plan, + { + PlanID: plan.PlanID, + DisplayName: plan.DisplayName, + Balance: LookUp(Topic.BalanceTable, PlanID = plan.PlanID).Balance + } + ) + + - kind: SetVariable + id: build_intro_message + variable: Topic.introMessage + value: ="Sure, I'll help you submit a time off request. Here's a form where you can choose the type of leave and dates you want off. Don't see the type of leave you want? [Book directly in Workday](" & Topic.WorkdayUrl & ")" + + - kind: SendActivity + id: intro_message + activity: "{Topic.introMessage}" + - kind: ConditionGroup id: need_inputs conditions: @@ -52,88 +218,194 @@ beginDialog: actions: - kind: AdaptiveCardPrompt id: collect_time_off - displayName: Ask date, hours and reason for time off + displayName: Ask time off type, start date, end date and reason card: |- ={ type: "AdaptiveCard", '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", - version: "1.3", + version: "1.5", body: [ { type: "TextBlock", - text: "Request Time Off", + text: "Book your time off", weight: "Bolder", size: "Medium", wrap: true, color: "Default" }, + { + type: "Container", + style: "emphasis", + bleed: true, + items: [ + { + type: "TextBlock", + text: "Available balance in hours as of " & Text(Topic.AsOfEffectiveDate, "mmmm d, yyyy"), + size: "Small", + weight: "Bolder", + wrap: true + }, + { + type: "ColumnSet", + columns: ForAll( + Topic.DisplayBalances, + { + type: "Column", + width: "stretch", + items: [ + { type: "TextBlock", text: DisplayName, size: "Small", wrap: true }, + { type: "TextBlock", text: Balance, weight: "Bolder", spacing: "Small" } + ] + } + ) + } + ] + }, { type: "TextBlock", - text: "Enter the date, hours, and reason for your time off request.", - wrap: true + text: "Fields marked with * are required.", + wrap: true, + spacing: "Medium", + size: "Small" }, { - type: "Input.ChoiceSet", - id: "timeOffType", - label: "Type of Time Off", - style: "compact", - isRequired: true, - errorMessage: "Please select a type.", - choices: [ - { title: "Floating Holiday", value: "Floating_Holiday_Hours" }, - { title: "Vacation", value: "Vacation_Hours" }, - { title: "Sick", value: "Sick_Hours" } - ] - }, + type: "ColumnSet", + columns: [ + { + type: "Column", + width: "stretch", + items: [ + { + type: "Input.ChoiceSet", + id: "timeOffType", + label: "Type of time off", + style: "compact", + value: Topic.MappedTimeOffType, + isRequired: true, + errorMessage: "Please select a type.", + choices: [ + { title: "Vacation", value: "Vacation_Hours" }, + { title: "Floating holiday", value: "Floating_Holiday_Hours" }, + { title: "Sick leave", value: "Sick_Hours" }, + { title: "I don't see my leave type listed", value: "NOT_LISTED" } + ] + } + ] + } + ] + }, { type: "Input.Date", - id: "timeOffDate", - label: "Date", + id: "startDate", + label: "Start date", + value: If(IsBlank(Topic.InputStartDate), "", Text(Topic.InputStartDate, "yyyy-MM-dd")), isRequired: true, - errorMessage: "Please select a date.", - value: If(IsBlank(Topic.TimeOffDate) || Topic.TimeOffDate < Today(), "",Text(Topic.TimeOffDate,"yyyy-MM-dd")) + errorMessage: "Please select a start date." }, { - type: "Input.Number", - id: "numberOfHours", - label: "Number of Hours", - placeholder: "Enter hours (e.g., 8, 4, 2)", + type: "Input.Date", + id: "endDate", + label: "End date", + value: If(IsBlank(Topic.InputEndDate), "", Text(Topic.InputEndDate, "yyyy-MM-dd")), isRequired: true, - errorMessage: "Please enter the number of hours.", - min: 0.5, - max: 24, - value: If(IsBlank(Topic.NumberOfHours), 8, Topic.NumberOfHours) + errorMessage: "Please select an end date." }, { - type: "Input.Text", - id: "reasonText", - placeholder: "Reason for time off", - label: "Reason", + type: "Input.Number", + id: "hoursPerDay", + label: "Hours per day", + min: 1, + max: 24, + value: If(IsBlank(Topic.InputHoursPerDay), 8, Value(Topic.InputHoursPerDay)), isRequired: true, - maxLength: 500, - errorMessage: "Reason is required.", - value: Topic.ReasonText + errorMessage: "Please enter hours per day." + }, + { + type: "ColumnSet", + spacing: "Medium", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "ActionSet", + actions: [ + { type: "Action.Submit", title: "Submit", id: "Submit" }, + { type: "Action.Submit", title: "Cancel", id: "Cancel" } + ] + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "stretch", + items: [], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "ColumnSet", + spacing: "None", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "Image", + url: Topic.WorkdayIconUrl, + style: "RoundedCorners", + size: "Small", + height: "20px", + width: "20px" + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "TextBlock", + text: "Workday", + size: "Small", + color: "#242424", + weight: "Bolder" + } + ], + verticalContentAlignment: "Center", + spacing: "Small" + } + ] + } + ], + verticalContentAlignment: "Center" + } + ] } ], - actions: [ - { type: "Action.Submit", title: "Submit" }, - { type: "Action.Submit", title: "Cancel", associatedInputs: "none" } - ] + actions: [] } output: binding: actionSubmitId: Topic.actionSubmitId - numberOfHours: Topic.numberOfHours - reasonText: Topic.reasonText - timeOffDate: Topic.timeOffDate + endDate: Topic.endDate + hoursPerDay: Topic.hoursPerDay + startDate: Topic.startDate timeOffType: Topic.timeOffType outputType: properties: actionSubmitId: String - numberOfHours: Number - reasonText: String - timeOffDate: String + endDate: String + hoursPerDay: String + startDate: String timeOffType: String - kind: ConditionGroup @@ -149,13 +421,128 @@ beginDialog: - kind: CancelAllDialogs id: cancel_all + # Handle "I don't see my leave type listed" selection + - kind: ConditionGroup + id: handle_not_listed + conditions: + - id: type_not_listed + condition: =Topic.timeOffType = "NOT_LISTED" + actions: + - kind: SetVariable + id: set_not_listed_message + variable: Topic.notListedMessage + value: ="Since your leave type isn't listed here, you can book directly in [Workday](" & Topic.WorkdayUrl & "), or let me know if you want to change your type." + + - kind: SendActivity + id: not_listed_msg + activity: "{Topic.notListedMessage}" + + - kind: CancelAllDialogs + id: end_not_listed + + # Validate end date >= start date + - kind: ConditionGroup + id: validate_dates + conditions: + - id: end_before_start + condition: =DateValue(Topic.endDate) < DateValue(Topic.startDate) + actions: + - kind: SendActivity + id: date_error_msg + activity: The end date can't be earlier than the start date. + + # Ask user if they want to try again + - kind: Question + id: ask_retry_dates + interruptionPolicy: + allowInterruption: false + variable: Topic.retryDateChoice + prompt: Would you like to try with different dates? + entity: BooleanPrebuiltEntity + + - kind: ConditionGroup + id: handle_retry_date_choice + conditions: + - id: user_wants_retry_dates + condition: =Topic.retryDateChoice = true + actions: + - kind: GotoAction + id: goto_form_on_date_retry + actionId: collect_time_off + elseActions: + - kind: SendActivity + id: end_on_no_date_retry + activity: No problem! Let me know if you need anything else. + + - kind: CancelAllDialogs + id: cancel_on_no_date_retry + + # ======================================================== + # V3: Build list of dates and pass to Plugin + # ======================================================== + + # Initialize date list (empty string) + - kind: SetVariable + id: init_date_list + variable: Topic.dateList + value: ="" + + # Initialize loop counter + - kind: SetVariable + id: init_iterator + variable: Topic.newIterator + value: =0 + + # Set up loop variable + - kind: SetVariable + id: set_iterator + variable: Topic.iterator + value: =Topic.newIterator + + # Calculate current date for this iteration + - kind: SetVariable + id: calc_current_date + variable: Topic.currentDate + value: =Text(DateAdd(DateValue(Topic.startDate), Topic.iterator, TimeUnit.Days), "yyyy-MM-dd") + + # Loop to build comma-separated date list + - kind: ConditionGroup + id: build_date_list_loop + conditions: + - id: should_continue_building + condition: =DateValue(Topic.currentDate) <= DateValue(Topic.endDate) + actions: + # Append date to list (with comma separator if not first) + - kind: SetVariable + id: append_date + variable: Topic.dateList + value: =If(Topic.dateList = "", Topic.currentDate, Topic.dateList & "," & Topic.currentDate) + + # Increment counter + - kind: SetVariable + id: increment_iterator + variable: Topic.newIterator + value: =Topic.newIterator + 1 + + # Continue loop + - kind: GotoAction + id: goto_build_loop + actionId: set_iterator + + # Total days is the count from loop + - kind: SetVariable + id: calc_total_days + variable: Topic.totalDays + value: =Topic.newIterator + + # Make single API call - Plugin will build XML entries from date list - kind: BeginDialog - id: execute_workday - displayName: Redirect to Workday System Get Common Execution + id: execute_workday_multiday + displayName: Submit Multi-Day Time Off Request input: binding: - parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{Time_Off_Date}"",""value"":""" & Topic.timeOffDate & """},{""key"":""{Comment}"",""value"":""" & Topic.reasonText & """},{""key"":""{Reason_ID}"",""value"":""" & Topic.timeOffType & """},{""key"":""{Hours}"",""value"":""" & Text(Topic.numberOfHours) & """}]}" - scenarioName: msdyn_HRWorkdayAbsenceEnterTimeOff_EnterTimeOffInfo + parameters: ="{""params"":[{""key"":""{Employee_ID}"",""value"":""" & Global.ESS_UserContext_Employee_Id & """},{""key"":""{timeoff_Dates}"",""value"":""" & Topic.dateList & """},{""key"":""{timeoff_Time_Off_Type}"",""value"":""" & Topic.timeOffType & """},{""key"":""{timeoff_Hours_Per_Day}"",""value"":""" & Topic.hoursPerDay & """},{""key"":""{timeoff_Comment}"",""value"":""ess generated time off request""}]}" + scenarioName: msdyn_HRWorkdayAbsenceEnterTimeOff_MultiDay dialog: msdyn_copilotforemployeeselfservicehr.topic.WorkdaySystemGetCommonExecution output: @@ -164,20 +551,191 @@ beginDialog: isSuccess: Topic.isSuccess workdayResponse: Topic.workdayResponse + # Parse error message for display + - kind: SetVariable + id: init_last_error + variable: Topic.lastErrorMessage + value: =If(Topic.isSuccess = true, "", Topic.errorResponse) + + - kind: SetVariable + id: parse_error_message + variable: Topic.friendlyErrorMessage + value: =IfError(Text(ParseJSON(Topic.lastErrorMessage).error.message), IfError(Text(ParseJSON(Topic.lastErrorMessage).message), Topic.lastErrorMessage)) + - kind: ConditionGroup id: report_result conditions: - - id: failed + - id: request_failed condition: =Topic.isSuccess = false actions: + # Use AI to generate a friendly, conversational error message + - kind: AnswerQuestionWithAI + id: generate_friendly_error + autoSend: false + variable: Topic.aiGeneratedError + userInput: =Topic.friendlyErrorMessage + additionalInstructions: Reframe the following Workday error message in a friendly way for an employee. Keep it to ONE short sentence describing what went wrong. Do NOT include suggestions or next steps. Do NOT apologize. Do NOT use technical jargon. Example output format - "The dates you selected conflict with an existing request." + + # Set final error message with fallback + - kind: SetVariable + id: set_final_error_message + variable: Topic.finalErrorMessage + value: =If(IsBlank(Topic.aiGeneratedError), "The dates you selected aren't available.", Topic.aiGeneratedError) + - kind: SendActivity - id: failure_msg - activity: An error occurred and your time off request was not submitted. + id: friendly_error_message + activity: "{Topic.finalErrorMessage}" + + # Ask user if they want to try again + - kind: Question + id: ask_retry + interruptionPolicy: + allowInterruption: false + variable: Topic.retryChoice + prompt: Would you like to try with different dates? + entity: BooleanPrebuiltEntity + + - kind: ConditionGroup + id: handle_retry_choice + conditions: + - id: user_wants_retry + condition: =Topic.retryChoice = true + actions: + - kind: GotoAction + id: goto_form_on_retry + actionId: collect_time_off + elseActions: + - kind: SendActivity + id: end_on_no_retry + activity: No problem! Let me know if you need anything else. + + - kind: CancelAllDialogs + id: cancel_on_no_retry elseActions: - kind: SendActivity - id: success_msg - activity: Your time off request has been submitted. + id: success_card + activity: + attachments: + - kind: AdaptiveCardTemplate + cardContent: |- + ={ + type: "AdaptiveCard", + '$schema': "http://adaptivecards.io/schemas/adaptive-card.json", + version: "1.5", + body: [ + { + type: "TextBlock", + text: "✅ Request submitted", + weight: "Bolder", + size: "Medium", + wrap: true + }, + { + type: "TextBlock", + text: "Your time off request has been successfully submitted and is now pending approval.", + wrap: true, + spacing: "Small" + }, + { + type: "Table", + spacing: "Medium", + showGridLines: false, + firstRowAsHeader: false, + columns: [ + { width: 1 }, + { width: 2 } + ], + rows: [ + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Type of time off", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Switch(Topic.timeOffType, "Vacation_Hours", "Vacation", "Floating_Holiday_Hours", "Floating holiday", "Sick_Hours", "Sick leave", Topic.timeOffType), wrap: true }] + } + ] + }, + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Time off date range", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Text(DateValue(Topic.startDate), "mmmm d, yyyy") & " to " & Text(DateValue(Topic.endDate), "mmmm d, yyyy") & " (" & Text(Topic.totalDays) & " days)", wrap: true }] + } + ] + }, + { + type: "TableRow", + cells: [ + { + type: "TableCell", + items: [{ type: "TextBlock", text: "Total hours", weight: "Bolder" }] + }, + { + type: "TableCell", + items: [{ type: "TextBlock", text: Text(Topic.totalDays * Value(Topic.hoursPerDay)) & " hours (" & Text(Topic.totalDays) & " x " & Topic.hoursPerDay & "-hour days)", wrap: true }] + } + ] + } + ] + }, + { + type: "Container", + horizontalAlignment: "Right", + items: [ + { + type: "ColumnSet", + columns: [ + { + type: "Column", + width: "auto", + items: [ + { + type: "Image", + url: Topic.WorkdayIconUrl, + style: "RoundedCorners", + size: "Small", + height: "20px", + width: "20px" + } + ], + verticalContentAlignment: "Center" + }, + { + type: "Column", + width: "auto", + items: [ + { + type: "TextBlock", + text: "Workday", + size: "Small", + color: "#242424", + weight: "Bolder" + } + ], + verticalContentAlignment: "Center", + spacing: "Small" + } + ] + } + ] + } + ], + actions: [] + } + + - kind: SendActivity + id: follow_up_msg + activity: Can I help you submit another request? - kind: CancelAllDialogs id: end_dialogs @@ -188,20 +746,21 @@ inputType: displayName: EmployeeName description: The name of the employee to fetch data for. type: String - - NumberOfHours: - displayName: NumberOfHours - description: Number of hours for time off + InputStartDate: + displayName: InputStartDate + description: The start date of the time off request. + type: DateTime + InputEndDate: + displayName: InputEndDate + description: The end date of the time off request. + type: DateTime + InputHoursPerDay: + displayName: InputHoursPerDay + description: The number of hours per day for the time off request. type: Number - - ReasonText: - displayName: ReasonText - description: Reason for time off (free text) + InputTimeOffType: + displayName: InputTimeOffType + description: The type of time off being requested. type: String - TimeOffDate: - displayName: TimeOffDate - description: Date of time off (yyyy-mm-dd) - type: DateTime - outputType: {} \ No newline at end of file