Как вложить данные из таблицы соединений в отношение «один (или несколько) ко многим» в knex.js? - PullRequest
1 голос
/ 09 июня 2019

Допустим, у меня есть пользователи, навыки и таблицы user_skill.

Как я могу сделать запрос, чтобы, если у меня было 3 пользователя, я получил 3 объекта (в массиве, очевидно), и для объекта каждого пользователя должен быть вложенный массив со всеми навыками?

Примерно так:

{
    "users": [
        {
            "name": "dusan",
            "id": 1,
            "facebook": "dusan's facebook",
            "skills": [
                {"name": "skill1", "id": 1}
                {"name": "skill2", "id": 2}
            ]
        },
        {
            "name": "jenny",
            "id": 2,
            "facebook": "jenny's facebook",
            "skills": [
                {"name": "skill1", "id": 1}
                {"name": "skill2", "id": 2}
            ]
        },
        {
            "name": "michael",
            "id": 3,
            "facebook": "michael's facebook",
            "skills": [
                {"name": "skill1", "id": 1}
                {"name": "skill2", "id": 2}
            ]
        },

    ]
}

Мой код:

await db("users")
  .join("user_skill",  "users.id", "skills.user_id")
  .join("skills", "skills.id", "user_skill.id")
  .select(
    "users.name as name",
    "users.id as id",
    "skills.name as skill",
    "skills.id as skill_id"
);

Приведенный выше код возвращает полностью плоский структурированный JSON, и я хочу иметь навыки, вложенные в объект пользователя.

{
    "users": [
        {
            "name": "dusan",
            "id": 1,
            "skill": "node.js",
            "skill_id": 1
        },
        {
            "name": "dusan",
            "id": 1,
            "skill": "php",
            "skill_id": 2
        },
        {
            "name": "dusan",
            "id": 1,
            "skill": "mongodb",
            "skill_id": 3
        },
        {
            "name": "jaca",
            "id": 2,
            "skill": "angular",
            "skill_id": 4
        },
        {
            "name": "jaca",
            "id": 2,
            "skill": "reactjs",
            "skill_id": 5
        },
    ]
}

Я понимаю, что так устроен SQL, чтобы возвращать плоскую структуру, но есть ли способ, которым я мог бы выполнить вложение связанных данных?

1 Ответ

0 голосов
/ 20 июня 2019

Другой разработчик предложил это решение, поэтому я решил поделиться им в надежде, что кто-то найдет его полезным:

returnedUsers = {
  "users": [{
      "name": "dusan",
      "id": 1,
      "skill": "node.js",
      "skill_id": 1
    },
    {
      "name": "dusan",
      "id": 1,
      "skill": "php",
      "skill_id": 2
    },
    {
      "name": "dusan",
      "id": 1,
      "skill": "mongodb",
      "skill_id": 3
    },
    {
      "name": "jaca",
      "id": 2,
      "skill": "angular",
      "skill_id": 4
    },
    {
      "name": "jaca",
      "id": 2,
      "skill": "reactjs",
      "skill_id": 5
    },
  ]
};

const groupBy = (array, key) =>
  array.reduce((a, c) => ({
    ...a,
    [c[key]]: [...a[c[key]] || [], c]
  }), {});

const uniques = (...values) =>
  Array.from(new Set([].concat(...values).filter(Boolean)));

const singularize = array =>
  array.length == 1 ? array[0] : array;

const singularizedUniques = (...values) =>
  singularize(uniques(...values));

const mergeCollect = array =>
  array.reduce((mergedObject, curentObject) =>
    Object.entries(curentObject).reduce((newObject, [k, v]) => ({
      ...newObject,
      [k]: singularizedUniques(newObject[k], v)
    }), mergedObject), {});

const groupByKey = (array, key) =>
  Object.fromEntries(Object.entries(groupBy(array, key)).map(([k, v]) => [k, mergeCollect(v)]));

console.log(groupByKey(returnedUsers.users, 'id'));
...