Индекс CouchDB для соединения точек между документами - PullRequest
0 голосов
/ 13 декабря 2018

У меня есть следующие документы:

{ _id: "123", type: "project", worksite_id: "worksite_1" }
{ _id: "456", type: "document", project_id: "123" }
{ _id: "789", type: "signature", document_id: "456" }

Моя цель - выполнить запрос и неизбежно выполнить отфильтрованную репликацию всех документов, связанных с worksite_id: worksite_1.

Пример:

  1. Поскольку этот проект имеет рабочую площадку, которую я ищу,
  2. документ имеет этот проект
  3. Подпись имеет этот документ

Я должен быть в состоянии получить все эти документы, если мне нужно все с этого рабочего места.

Обычно я просто добавляю worksite_id к своим type:document и type:signature.Однако рабочие места могут изменяться в проекте по разным причинам.

Мне было интересно, есть ли способ создать индекс или сделать что-то, о чем я не думаю, чтобы показать эти сходства.

Это кажется, что оно находится на правильном пути, но объяснение помещает документы в другие документы, где я просто хочу, чтобы они были отдельными.

Любая помощь будет принята с благодарностью.

Ответы [ 2 ]

0 голосов
/ 21 декабря 2018

Я думаю, у меня есть то, что вы ищете.

Вот некоторые данные:

{
  "docs": [
    {
        "_id": "123",
        "type": "project",
        "code": "p001"
    },
    {
        "_id": "1234",
        "type": "worksitelog",
        "documents": [
          {
            "timestamp": "20180921091501",
            "project_id": "123",
            "document_id": "457",
            "signature_id": "789"
          },
          {
            "timestamp": "20180921091502",
            "project_id": "123",
            "document_id": "457",
            "signature_id": "791"
          },
          {
            "timestamp": "20180921091502",
            "project_id": "123",
            "document_id": "458",
            "signature_id": "791"
          },
          {
            "timestamp": "20180921091502",
            "project_id": "123",
            "document_id": "456",
            "signature_id": "790"
          }
        ],
        "worksite_id": "worksite_2"
    },
    {
        "_id": "1235",
        "type": "worksitelog",
        "documents": [
          {
            "timestamp": "20180913101502",
            "project_id": "125",
            "document_id": "459",
            "signature_id": "790"
          }
        ],
        "worksite_id": "worksite_1"
    },
    {
        "_id": "124",
        "type": "project",
        "code": "p002"
    },
    {
        "_id": "125",
        "type": "project",
        "code": "p003"
    },
    {
        "_id": "456",
        "type": "document",
        "code": "d001",
        "project_id": "123",
        "worksite_id": "worksite_2"
    },
    {
        "_id": "457",
        "type": "document",
        "code": "d002",
        "project_id": "123",
        "worksite_id": "worksite_2"
    },
    {
        "_id": "458",
        "type": "document",
        "code": "d003",
        "project_id": "123",
        "worksite_id": "worksite_2"
    },
    {
        "_id": "459",
        "type": "document",
        "code": "d001",
        "project_id": "125",
        "worksite_id": "worksite_1"
    },
    {
        "_id": "789",
        "type": "signature",
        "user": "alice",
        "pubkey": "65ab64c64ed64ef41a1bvc7d1b",
        "code": "s001"
    },
    {
        "_id": "790",
        "type": "signature",
        "user": "carol",
        "pubkey": "tlmg90834kmn90845kjndf98734",
        "code": "s002"
    },
    {
        "_id": "791",
        "type": "signature",
        "user": "bob",
        "pubkey": "asdf654asdf6854awer654awer654eqr654wra6354f",
        "code": "s003"
    },
    {
        "_id": "_design/projDocs",
        "views": {
          "docsPerWorkSite": {
            "map": "function (doc) {\n  if (doc.type && ['worksitelog', 'document', 'project', 'signature'].indexOf(doc.type) > -1) {\n    if (doc.type == 'worksitelog') {\n      emit([doc.worksite_id, 0], null);\n      for (var i in doc.documents) {\n        emit([doc.worksite_id, Number(i)+1, 'p'], {_id: doc.documents[i].project_id});\n        emit([doc.worksite_id, Number(i)+1, 'd'], {_id: doc.documents[i].document_id});\n        emit([doc.worksite_id, Number(i)+1, 's'], {_id: doc.documents[i].signature_id});\n      }\n    }\n  }\n}"
          }
        },
        "language": "javascript"
    }
  ]
}

Сохраните эти данные на диск как stackoverflow_53752001.json.

Используйте Fauxtonдля создания базы данных с именем stackoverflow_53752001.

Вот скрипт bash для загрузки данных из файла stackoverflow_53752001.json into the database stackoverflow_53752001`.Вам нужно будет отредактировать первые три параметра, очевидно.Исправьте это, затем вставьте его в окно терминала (Unix):

USRID="you";
USRPWD="yourpwd";
HOST="yourdb.yourpublic.work";

COUCH_DATABASE="stackoverflow_53752001";
FILE="stackoverflow_53752001.json";
#
COUCH_URL="https://${USRID}:${USRPWD}@${HOST}";
FULL_URL="${COUCH_URL}/${COUCH_DATABASE}";
curl -H 'Content-type: application/json' -X POST "${FULL_URL}/_bulk_docs"  -d @${FILE};

В Fauxton выберите базу данных stackoverflow_53752001, а затем в левом меню выберите «Проектные документы» >> «projDocs»>> «Представления» >> «docsPerWorkSite».

Вы увидите такие данные:

{"total_rows":17,"offset":0,"rows":[
  {"id":"1235","key":["worksite_1",0],"value":null},
  {"id":"1235","key":["worksite_1",1,"d"],"value":{"_id":"459"}},
          :                     :
          :                     :
  {"id":"1234","key":["worksite_2",4,"p"],"value":{"_id":"123"}},
  {"id":"1234","key":["worksite_2",4,"s"],"value":{"_id":"790"}}
]}

Если затем нажать кнопку «Параметры» в правом верхнем углу,вы получите лист опций для изменения необработанного запроса.Выбор:

  • "Включить документы"
  • "Между ключами"
    • "Ключ запуска": ["worksite_1", 0]
    • "Клавиша завершения ": [" worksite_1 ", 9999]

Нажмите" Run Query ", и вы должны увидеть:

{"total_rows":17,"offset":0,"rows":[
  {"id":"1235","key":["worksite_1",0],"value":null,"doc":{"_id":"1235","_rev":"1-de2b919591c70f643ce1005c18da1c54","type":"worksitelog","documents":[{"timestamp":"20180913101502","project_id":"125","document_id":"459","signature_id":"790"}],"worksite_id":"worksite_1"}},
  {"id":"1235","key":["worksite_1",1,"d"],"value":{"_id":"459"},"doc":{"_id":"459","_rev":"1-5422628e475bab0c14e5722a1340f561","type":"document","code":"d001","project_id":"125","worksite_id":"worksite_1"}},
  {"id":"1235","key":["worksite_1",1,"p"],"value":{"_id":"125"},"doc":{"_id":"125","_rev":"1-312dd8a9dd432168d8608b7cd9eb92cd","type":"project","code":"p003"}},
  {"id":"1235","key":["worksite_1",1,"s"],"value":{"_id":"790"},"doc":{"_id":"790","_rev":"1-be018df4ecdf2e6add68a2758b9bd12a","type":"signature","user":"carol","pubkey":"tlmg90834kmn90845kjndf98734","code":"s002"}}
]}

Если вы затем изменитес клавишами начала и конца ["worksite_2", 0] и ["worksite_2", 9999] вы увидите данные для второго рабочего сайта.

Чтобы это работало, каждый раз, когда вы записываете новый документ и подпись в базу данных, выВам нужно будет:

  1. подготовить объект { "timestamp": "20180921091502", "project_id": "123", "document_id": "457", "signature_id": "791" }
  2. получить соответствующую запись журнала рабочей площадки
  3. добавить объект в массив documents
  4. положить обратно измененную запись журнала рабочего сайта

Я предположил, что на каждый документ приходится несколько подписей, поэтому вам придется записывать запись журнала для каждого из них.Если это становится слишком большим, вы можете изменить worksite_id на что-то вроде worksite_1_201812, что даст один журнал на каждый рабочий сайт в месяц без нарушения логики запросов, я думаю.

0 голосов
/ 19 декабря 2018

Функция карты рассматривает только один документ за раз, поэтому, если этот документ не знает о других документах, вы не можете связать их вместе.Ваша структура подразумевает объединение трех таблиц в терминах SQL.

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

function (doc) {
  if (doc && doc.type && doc.type === "signature" && doc.document_id) {
    emit(doc.document_id, {_id: doc.document_id})
  }
}

и, используя ту же технику, связывает проекты с документами, но вы не можете связать все три.

...