Obtén versiones específicas de los recursos

Cada recurso tiene un campo de versión que se modifica cada vez que lo hace el recurso: el campo etag. Las Etags son una parte estándar de HTTP y se admiten en la API de Calendario en dos casos:

  • sobre las modificaciones de los recursos para garantizar que no haya habido más operaciones de escritura en este mientras tanto (modificación condicional)
  • en la recuperación de recursos para recuperar solo los datos de estos si estos cambiaron (recuperación condicional)

Modificación condicional

Si deseas actualizar o borrar un recurso solo si no cambió desde la última vez que lo recuperaste, puedes especificar un encabezado If-Match que contenga el valor del ETag de la recuperación anterior. Esto es muy útil para evitar la pérdida de modificaciones en los recursos. Los clientes tienen la opción de volver a recuperar el recurso y aplicar los cambios.

Si la entrada (y su ETag) no cambió desde la última recuperación, la modificación se realizará correctamente y se mostrará la nueva versión del recurso con la ETag nueva. De lo contrario, obtendrás un código de respuesta 412 (error de condición previa).

En el fragmento de código de muestra que aparece a continuación, se muestra cómo realizar modificaciones condicionales con la biblioteca cliente de Java.

  private static void run() throws IOException {
    // Create a test event.
    Event event = Utils.createTestEvent(client, "Test Event");
    System.out.println(String.format("Event created: %s", event.getHtmlLink()));

    // Pause while the user modifies the event in the Calendar UI.
    System.out.println("Modify the event's description and hit enter to continue.");
    System.in.read();

    // Modify the local copy of the event.
    event.setSummary("Updated Test Event");

    // Update the event, making sure that we don't overwrite other changes.
    int numAttempts = 0;
    boolean isUpdated = false;
    do {
      Calendar.Events.Update request = client.events().update("primary", event.getId(), event);
      request.setRequestHeaders(new HttpHeaders().setIfMatch(event.getEtag()));
      try {
        event = request.execute();
        isUpdated = true;
      } catch (GoogleJsonResponseException e) {
        if (e.getStatusCode() == 412) {
          // A 412 status code, "Precondition failed", indicates that the etag values didn't
          // match, and the event was updated on the server since we last retrieved it. Use
          // {@link Calendar.Events.Get} to retrieve the latest version.
          Event latestEvent = client.events().get("primary", event.getId()).execute();

          // You may want to have more complex logic here to resolve conflicts. In this sample we're
          // simply overwriting the summary.
          latestEvent.setSummary(event.getSummary());
          event = latestEvent;
        } else {
          throw e;
        }
      }
      numAttempts++;
    } while (!isUpdated && numAttempts <= MAX_UPDATE_ATTEMPTS);

    if (isUpdated) {
      System.out.println("Event updated.");
    } else {
      System.out.println(String.format("Failed to update event after %d attempts.", numAttempts));
    }
  }

Recuperación condicional

Si deseas recuperar un recurso solo si cambió desde la última vez que lo recuperaste, puedes especificar un encabezado If-None-Match que contenga el valor de la etag de la recuperación anterior. Si la entrada (y, por lo tanto, su ETag) cambió desde la última recuperación, se mostrará la versión nueva del recurso con la ETag nueva. De lo contrario, obtendrás un código de respuesta 304 (Not Modified).

En el fragmento de código de muestra que aparece a continuación, se demuestra cómo realizar una recuperación condicional con la biblioteca cliente de Java.

  private static void run() throws IOException {
    // Create a test event.
    Event event = Utils.createTestEvent(client, "Test Event");
    System.out.println(String.format("Event created: %s", event.getHtmlLink()));

    // Pause while the user modifies the event in the Calendar UI.
    System.out.println("Modify the event's description and hit enter to continue.");
    System.in.read();

    // Fetch the event again if it's been modified.
    Calendar.Events.Get getRequest = client.events().get("primary", event.getId());
    getRequest.setRequestHeaders(new HttpHeaders().setIfNoneMatch(event.getEtag()));
    try {
      event = getRequest.execute();
      System.out.println("The event was modified, retrieved latest version.");
    } catch (GoogleJsonResponseException e) {
      if (e.getStatusCode() == 304) {
        // A 304 status code, "Not modified", indicates that the etags match, and the event has
        // not been modified since we last retrieved it.
        System.out.println("The event was not modified, using local version.");
      } else {
        throw e;
      }
    }
  }