- Subscribe to RSS Feed
- Mark Topic as New
- Mark Topic as Read
- Float this Topic for Current User
- Bookmark
- Subscribe
- Printer Friendly Page
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

List.Generate to recreate PATH dax command in power query m
I have a table that lists employees and their parent employees.
I want to add a column that contains a concatenated list of all parent employees in the hierarchy (similar to the PATH function in DAX). List.Generate seems like it could fit my needs.
I tried to create the column using List.Generate as below:
let
table = Table.Buffer(#"Grouped Rows")
in
List.Generate(
() => [emp = [employeeID], super = [supervisorID]],
each [super] <> null,
each [emp = [super], super = (a)=> List.Max(Table.SelectRows(table, (b)=> [employeeID] = (a)[super])[supervisorID])],
each [emp]
)
As I understand it, the code I wrote is supposed to do the following:
- Define two variables and their starting points (emp as the current row's employee, super as the parent employee)
- Check whether the current state of [super] is null. If so, then end loop. If not, then proceed.
- Replace the value of [emp] with the value of [super], and set the value of [super] to the current [super's] parent employee.
- Add the current value of [emp] to the list.
- Repeat until [super] is null.
This works fine until the function is supposed to take over, at which point the list just says "function" for every record (see screenshot). Is there something I'm missing? Does List.Generate simply not allow functions in this area?
Thanks in advance!
Solved! Go to Solution.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

Hello, @sean_cochran List.Generate generates list of smth - scalars, records, lists, maybe list of functions or list of whatever you want it to generate. In your particular example a list of (a) => ... functions is generated. Calculate something. While you are working on your own solution may I suggest to consider mine? This was entertaining. The idea is to grab bosses from bottom to top levels first, then go over this list again to add bosses to each employee step by step.
let
Source = your_table,
// make an initial record ee as fields and and bosses as values
rec = Record.FromList(Source[supervisorID], Source[employeeID]),
// this function creates next level of supervisors
fx = (r as record) =>
[d = List.Difference(Record.FieldNames(r[next]), Record.FieldValues(r[next])),
out = [lst = Record.SelectFields(r[next], d), next = Record.RemoveFields(r[next], d), flag = Record.FieldCount(r[next]) > 0]][out],
// this List.Generate goes from bottom level to top and creates lists of supervisors
gena =
List.Generate(
() => fx([next = rec]),
(x) => x[flag],
(x) => fx(x),
(x) => x[lst]
),
p = {1..(List.Count(gena) - 1)},
// here we go from bottom to top and add supervisors step by step
acc =
List.Accumulate(
p,
Table.FromColumns({Record.FieldNames(gena{0}), List.Transform(Record.FieldValues(gena{0}), each {_})}, {"employee", "super"}),
(s, c) =>
[g = gena{c},
employees = Record.FieldNames(g),
supervisors = Record.FieldValues(g),
tbl = Table.FromColumns({employees, List.Transform(supervisors, each {_})}, {"employee", "super"}),
add_super = Table.TransformColumns(s, {"super", (x) => x & Record.FieldValues(Record.SelectFields(g, x, MissingField.Ignore))}),
add_level = Table.Combine({add_super, tbl})][add_level]
),
out = Table.TransformColumns(acc, {"super", (w) => Text.Combine(w, ", ")})
in
out
providing that you have 1 supervisor per each employee.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

I've applied the solution from @AlienSx , but when used in a dataflow, it seems to cause issues with nightly scheduled dataflow refreshes: "Error: Expression.Error: We expected a FieldsSelector value. <ccon>{null}</ccon>. RootActivityId = c9335db1-3174-400f-ae83-3286d034d25c.Param1 = Expression.Error: We expected a FieldsSelector value. <ccon>{null}</ccon> Request ID: d871921e-4364-488b-9744-036185561172."
I can repost this elsewhere if that's required, but since the issue is specific to this solution, I wanted to list it here at least. Does anyone have any ideas? Here is the query code:
let
Source = #"Employment Details",
#"Filtered Rows" = Table.SelectRows(Source, each [employeeStatusCode] = "A" and [companyID] <> "A76BG"),
#"Grouped Rows" = Table.Group(#"Filtered Rows", {"employeeID"}, {{"supervisorID", each List.Max([supervisorID]), type nullable text}}),
employees = Table.Buffer(#"Grouped Rows"),
// make an initial record ee as fields and and bosses as values
rec = Record.FromList(employees[supervisorID], employees[employeeID]),
// this function creates next level of supervisors
fx = (r as record) =>
[d = List.Difference(Record.FieldNames(r[next]), Record.FieldValues(r[next])),
out = [lst = Record.SelectFields(r[next], d), next = Record.RemoveFields(r[next], d), flag = Record.FieldCount(r[next]) > 0]][out],
// this List.Generate goes from bottom level to top and creates lists of supervisors
gena = List.Generate(
() => fx([next = rec]),
(x) => x[flag],
(x) => fx(x),
(x) => x[lst]
),
p = {1..(List.Count(gena) - 1)},
// here we go from bottom to top and add supervisors step by step
acc = List.Accumulate(
p,
Table.FromColumns({Record.FieldNames(gena{0}), List.Transform(Record.FieldValues(gena{0}), each {_})}, {"employee", "super"}),
(s, c) =>
[g = gena{c},
employees = Record.FieldNames(g),
supervisors = Record.FieldValues(g),
tbl = Table.FromColumns({employees, List.Transform(supervisors, each {_})}, {"employee", "super"}),
add_super = Table.TransformColumns(s, {"super", (x) => x & Record.FieldValues(Record.SelectFields(g, x, MissingField.Ignore))}),
add_level = Table.Combine({add_super, tbl})][add_level]
),
#"Expanded super" = Table.ExpandListColumn(acc, "super"),
#"Filtered rows 1" = Table.SelectRows(#"Expanded super", each [super] <> ""),
#"Transform columns" = Table.TransformColumnTypes(#"Filtered rows 1", {{"super", type text}}),
#"Replace errors" = Table.ReplaceErrorValues(#"Transform columns", {{"super", null}})
in
#"Replace errors"
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

This seems to work perfectly, but I can't claim to fully understand it. I'll be studying this for a while! Thank you for your help.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

Hello, @sean_cochran List.Generate generates list of smth - scalars, records, lists, maybe list of functions or list of whatever you want it to generate. In your particular example a list of (a) => ... functions is generated. Calculate something. While you are working on your own solution may I suggest to consider mine? This was entertaining. The idea is to grab bosses from bottom to top levels first, then go over this list again to add bosses to each employee step by step.
let
Source = your_table,
// make an initial record ee as fields and and bosses as values
rec = Record.FromList(Source[supervisorID], Source[employeeID]),
// this function creates next level of supervisors
fx = (r as record) =>
[d = List.Difference(Record.FieldNames(r[next]), Record.FieldValues(r[next])),
out = [lst = Record.SelectFields(r[next], d), next = Record.RemoveFields(r[next], d), flag = Record.FieldCount(r[next]) > 0]][out],
// this List.Generate goes from bottom level to top and creates lists of supervisors
gena =
List.Generate(
() => fx([next = rec]),
(x) => x[flag],
(x) => fx(x),
(x) => x[lst]
),
p = {1..(List.Count(gena) - 1)},
// here we go from bottom to top and add supervisors step by step
acc =
List.Accumulate(
p,
Table.FromColumns({Record.FieldNames(gena{0}), List.Transform(Record.FieldValues(gena{0}), each {_})}, {"employee", "super"}),
(s, c) =>
[g = gena{c},
employees = Record.FieldNames(g),
supervisors = Record.FieldValues(g),
tbl = Table.FromColumns({employees, List.Transform(supervisors, each {_})}, {"employee", "super"}),
add_super = Table.TransformColumns(s, {"super", (x) => x & Record.FieldValues(Record.SelectFields(g, x, MissingField.Ignore))}),
add_level = Table.Combine({add_super, tbl})][add_level]
),
out = Table.TransformColumns(acc, {"super", (w) => Text.Combine(w, ", ")})
in
out
providing that you have 1 supervisor per each employee.
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

Can you clarify a few things for me?
I'm used to seeing [next] used to reference table columns that have already been defined. In your solution, I can't see where the column [next] has been defined, but you refer to it in the function that creates the next level of supervisors:
fx = (r as record) =>
[d = List.Difference(Record.FieldNames(r[next]), Record.FieldValues(r[next])),
out = [lst = Record.SelectFields(r[next], d), next = Record.RemoveFields(r[next], d), flag = Record.FieldCount(r[next]) > 0]][out],
Maybe I'm misunderstanding something. I'm still trying to parse what you did here.
Thank you again for your help!
- Mark as New
- Bookmark
- Subscribe
- Mute
- Subscribe to RSS Feed
- Permalink
- Report Inappropriate Content

Nevermind! I think I got it. You told the function to expected a record "r" that contains the field "next", which is itself a varible, and you feed employee records into it via the list.generate statement. Still not sure I understand all the details, but I think I have the keys to figure it out now.

Helpful resources
Join us at the Microsoft Fabric Community Conference
March 31 - April 2, 2025, in Las Vegas, Nevada. Use code FABINSIDER for a $400 discount!
Power BI Monthly Update - February 2025
Check out the February 2025 Power BI update to learn about new features.

Subject | Author | Posted | |
---|---|---|---|
06-27-2023 03:38 AM | |||
06-08-2022 07:08 AM | |||
12-28-2022 08:03 PM | |||
06-05-2023 05:09 AM | |||
04-04-2024 03:30 AM |
User | Count |
---|---|
33 | |
32 | |
20 | |
15 | |
13 |
User | Count |
---|---|
20 | |
18 | |
15 | |
10 | |
10 |