Fusionar

Un registro OCDS une todos las entregas disponibles para un proceso de contrataciones en un momento dado, y puede incluir:

  • una entrega compilada, que expresa el estado actual del proceso de contratación, mostrando solamente los valores más recientes de los campos

  • una entrega versionada, que expresa todos los estados históricos del proceso de contratación, mostrando los valores de los campos en el tiempo

Fusionar es el proceso de combinar entregas individuales en una entrega compilada o versionada, descrito con más detalle abajo. A grandes rasgos:

  • Una entrega compilada se crea tomando solamente los valores más recientes de los campos de las entregas de un proceso de contratación específico.

  • Una entrega versionada se crea tomando todos los valores de los campos de las entregas en un proceso de contratación particular, copiando los metadatos sobre la entrega del cual se obtienen, y poniéndolos en orden cronológico.

Ejemplo Práctico

Un agencia de compras públicas publica un release para anunciar una oportunidad en Enero 1, en la cual el valor total estimado de la compra es $1,000. En Enero 31, publican un release para corregir la información, en el cual se expande la descripción de la compra. En Febrero 5, la agencia publica un release con una enmienda a la oportunidad, en la cual el valor total estimado de la compra ha aumentado a $2,000.

La agencia decide adjudicar la oportunidad a dos de los ofertantes. El 1 de Marzo, la agencia publica una entrega anunciando que se adjudicó un contrato de $750 a la Compañía A. El 3 de Marzo, la agencia publica una entrega anunciando que se adjudicó la compañía B un contrato de $750.

A través de estas entregas individuales, la agencia da datos en tiempo real sobre el proceso de contrataciones.

En cada entrega la agencia actualiza el registro, que combina todos las entregas a la fecha. En el registro final:

  • El release compilado contiene toda la información sobre la oportunidad y adjudicaciones, utilizando el mismo esquema que utiliza una entrega.

  • El release versionado hace más fácil ver como la descripción y el valor estimado total cambiaron con el tiempo.

{
    "ocid": "ocds-213czf-000-00002",
    "id": "ocds-213czf-000-00002-01-tender",
    "date": "2016-01-01T09:30:00Z",
    "language": "en",
    "tag": [
        "tender"
    ],
    "initiationType": "tender",
    "parties": [ ... ],
    "buyer": {...},
    "tender": {
        "id": "ocds-213czf-000-00002-01-tender",
        "title": "Data merging tool",
        "description": "Data merge tool.",
        "status": "active",
        "value": {
            "amount": 1000,
            "currency": "USD"
        },
        "procurementMethod": "open",
        "awardCriteria": "bestProposal",
        "submissionMethod": [
            "electronicSubmission"
        ],
        "tenderPeriod": {
            "startDate": "2016-01-31T09:00:00Z",
            "endDate": "2016-02-15T18:00:00Z"
        },
        "awardPeriod": {
            "startDate": "2016-04-01T00:00:00Z",
            "endDate": "2016-06-01T23:59:59Z"
        }
    }
}

Especificaciones de fusión (merging)

Campos descartados

En el esquema de entrega, "omitWhenMerged": true se declara en campos que se deben descartar cuando se hace la fusión. Estos actualmente son: id, date y tag.

  • Para un release compilado:

    • Tanto los campos como los valores se descartan porque son metadatos sobre entregas individuales; los releases compilados reemplazan estos con sus propios metadatos.

  • Para un release versionado:

    • Los campos se descartan, pero sus valores se mueven, como se describe abajo, para indicar de cuáles entregas se obtiene el valor del campo.

Si omitWhenMerged es false, ignórelo.

Nota

El release compilado actualmente usa el mismo esquema que la entrega OCDS, lo cual significa que los campos id, date y tag son requisitos en un release compilado. Invitamos a discutir sobre si es necesario hacer cambios a estos requisitos en un esquema separado de release compilado en el issue #330, y sobre cómo identificar y poner fecha a releases compilados y versionados en el issue #834.

Mientras tanto, una solución intermedia es poner tag igual a ["compiled"], date con la fecha de la entrega más reciente, y id igual a {ocid}-{date}, como en la referencia de implementación de la rutina de fusión (merge).

Valores versionados

Para convertir los valores de un campo de una entrega a un versioned value, debe:

  1. Crear un objeto JSON vacío

  2. Ajustar los campos releaseID, releaseDate y releaseTag a los valores id, date y tag de la entrega

  3. Ajustar el valor del campo (value) al valor del campo en la entrega

Así, un versioned value describe los valores de un campo en una entrega específica.

Por ejemplo, en el ejemplo de trabajo anterior, el valor estimado de compras fue $1,000 en una entrega (tender/value/amount fue 1000). Siguiendo los pasos de arriba, el valor versionado es:

{
  "releaseID": "ocds-213czf-000-00002-01-tender",
  "releaseDate": "2016-01-01T09:30:00Z",
  "releaseTag": [
    "tender"
  ],
  "value": 1000
}

En un versioned release, con algunas excepciones, un valor del campo se reemplaza con una lista de valores versionados, los cuales deben estar en orden cronológico ordenados por releaseDate.

Por ejemplo, en el ejemplo de trabajo anterior, el valor estimado fue $1,000 en una entrega publicada el 1 de Enero de 2016 y después $2,000 en una entrega publicada el 5 de Febrero de 2016. En un release versionado, esto se serializa de la siguiente manera:

{
    "amount": [
       {
            "releaseID": "ocds-213czf-000-00002-01-tender",
            "releaseDate": "2016-01-01T09:30:00Z",
            "releaseTag": [ ... ],
            "value": 1000
        },
       {
            "releaseID": "ocds-213czf-000-00002-01-tender-amendment",
            "releaseDate": "2016-02-05T10:30:00Z",
            "releaseTag": [ ... ],
            "value": 2000
        }
    ],
    "currency": [ ... ]
}

La estructura del release versionado se describe de acuerdo al esquema de release versionado; note que el valor del campo ocid no está versionado.

Rutina de fusión (merge)

Para crear un release compilado o versionado, debe de:

  1. Obtener todas las entregas con el mismo valor ocid

  2. Ordenar las entregas en orden cronológico por fecha (date)

  3. Crear un objeto JSON vacío para releases compilados o versionados

  4. Fusionar cada entrega (input), en orden, en el objeto JSON (output), de la siguiente manera:

Valores de objeto

La entrega es un objeto por si mismo, por lo que este caso se encuentra primero.

Si el objeto esta vacío en input, no debe hacer nada. Por cada campo dentro del objeto en input:

  • Para un release compilado:

    • Si el campo en input tiene el valor null, quite el campo del objeto en output, si está presente

    • Si el campo no está en output, añada el campo al objeto en output y establézcalo al valor presente en input

    • Si el campo está en output, fusione los valores de los campos en output e input de acuerdo a la rutina de fusión

  • Para un release versionado:

    • Fusione los valores de los campos en output e input de acuerdo a la rutina de fusión; si hay un resultado, agregue el campo al objeto en output si no esta agregado ya, y establezca su valor al resultado

Valores literales

Si el valor input no es un objeto ni una lista, entonces:

  • Para un release compilado:

    • Si el valor input es diferente al valor output, reemplace el valor output con el valor input

  • Para un release versionado:

    • Si no hay valor output, establezca el valor output como una lista JSON vacía, convierta el valor output a un valor versionado, y adjúntelo a la nueva lista de valores versionados en output

    • Si el valor input es diferente al valor del campo value del valor versionado más reciente en output, convierta el valor input a un valor versionado, y adjúntelo a la lista de valores versionados en output

Valores de lista

Si la lista input contiene algo más que objetos, trate la lista como un valor literal. De otra manera, hay dos sub rutinas para listas de objetos: fusión de lista completa y fusión de identificadores

Fusionar la lista completa

Una lista input debe tratarse como un valor literal si el campo correspondiente en una copia desreferenciada del esquema de entrega tiene "array" como type y si cualquiera de las siguientes son verdaderas:

  • El campo tiene "wholeListMerge": true`

  • El campo establece items/type, y tiene cualquier cosa menos "object" en su items/type

  • El campo tiene "object" en items/type, establece items/properties, y no tiene el campo id en items/properties

Fusión de identificador

Este caso se encuentra si las condiciones anteriores no se cumplen. Si la lista está vacía en input, no haga nada. Para cada objeto en la lista input:

  • Para un release compilado:

    • Si hay un objeto en la lista en output con el mismo valor de id en el objeto input, fusione los objetos iguales en input y output de acuerdo a la rutina de fusión

    • De otra manera, adjunte el objeto en input a la lista en output

  • Para un release versionado:

    • Si hay un objeto en la lista en output con el mismo valor de id que en el objeto en input, fusione los objetos iguales en input y output de acuerdo a la rutina de fusión, menos el campo id, que no está versionado y se debe mantener igual

    • De otra forma, fusione un objeto JSON vacío y el objeto en input de acuerdo a la rutina de fusión menos el campo id, que no está versionado y se debe mantener igual, y adjunte el resultado a la lista en output

Nota

En ese caso, para quitar un objeto de la lista, necesitas establecer cada uno de sus campos a null. Invitamos a discutir sobre cómo quitar objetos de listas en el issue #232.

Nota

En el esquema de entrega, "versionId": true se declara en los campos id que deben estar versionados. Esto es por conveniencia y debería retirarse en versiones futuras de OCDS (vea el issue #812). Si se declara "versionId": true en el campo id de un objeto en una lista, es ignorado. "versionId": false no tiene ningún significado y es ignorado.

Implementación de referencia

Una implementación de referencia de la rutina de fusión en Python está disponible en GitHub. Recomendamos fuertemente que, para cualquier re-implementación, lea el código comentado y use los casos de test, para asegurar que sea correcto.