common/src/sessionAnnotation.js

/*
Copyright 2020 Adobe. All rights reserved.
This file is licensed to you under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. You may obtain a copy
of the License at http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under
the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
OF ANY KIND, either express or implied. See the License for the specific language
governing permissions and limitations under the License.
*/

import * as kit from '@adobe/griffon-toolkit';
import schema from '../schemas/sessionAnnotation.json';

/**
 * Contains constants and functions for a Session Annotation Object.
 *
 * The structure for a Session Annotation Object is as follows:
 * ```
 * {
 *   payload: <object>,
 *   namespace: <string>,
 *   uuid: <string>,
 * }
 * ```
 *
 * @namespace sessionAnnotation
 */

/**
 * Paths for the keys on a Session Annotation Object
 *
 * @enum {string}
 */
const path = {
  /** An object with custom data describing the annotation.<br />Path is `payload`. */
  payload: 'payload',

  /** This is to scope annotations and prevent overwrites from other plugins. The type is usually deteremined by the plugin writing the annotation.<br />Path is `namespace`. */
  namespace: 'namespace',

  /** Uniquely identifies each annotation.<br />Path is `uuid`. */
  uuid: 'uuid'
};

/**
 * Describes the number of parents this object has based off schema references. When checking for matches for example, we want to
 * use a schema that is more specific over a more generic schema
 *
 * @constant
 */
const parentDepth = 0;

/**
 * A label that can be used when describing this object
 */
const label = 'Session Annotation Object';

/**
 * A grouping for this object
 */
const group = 'undefined';

/**
 * Retrieves a value from the object. You can provide either a path or an alias.
 *
 * @function
 * @param {string} alias Path or alias
 * @param {*} data Data to search
 * @returns {*}
 */
const get = (alias, data) => {
  const func = (data2) => kit.search(path[alias] || alias, data2);
  if (!data) { return func; }
  return func(data);
};

/**
 * Returns the `payload` from the Session Annotation Object.
 * This is the an object with custom data describing the annotation.
 *
 * Path is `payload`.
 *
 * @function
 * @param {object} source The Session Annotation Object instance
 * @returns {object}
 */
const getPayload = kit.search(path.payload);

/**
 * Returns the data using the specified path from the payload
 * of the Session Annotation Object.
 *
 * @function
 * @param {...string} path key in object
 * @param {object} source The Session Annotation Object instance
 * @returns {*}
 */
const getPayloadKey = kit.curry(
  (searchPath, source) => kit.search(`${path.payload}.${searchPath}`, source)
);

/**
 * Returns the `namespace` from the Session Annotation Object.
 * This is the this is to scope annotations and prevent overwrites from other plugins. The type is usually deteremined by the plugin writing the annotation.
 *
 * Path is `namespace`.
 *
 * @function
 * @param {object} source The Session Annotation Object instance
 * @returns {string}
 */
const getNamespace = kit.search(path.namespace);

/**
 * Returns the `uuid` from the Session Annotation Object.
 * This is the uniquely identifies each annotation.
 *
 * Path is `uuid`.
 *
 * @function
 * @param {object} source The Session Annotation Object instance
 * @returns {string}
 */
const getUuid = kit.search(path.uuid);

/**
 * Generates a Session Annotation Object with the const values set.
 * Can be useful in testing.
 * Can provide additional data by providing a flat object of paths and values.
 *
 * @function
 * @param {...Function} input Overrides
 * @returns {object}
 */
const make = kit.expand;

/**
 * Generates a Session Annotation Object with some default values set.
 * Can be useful in testing.
 * Can override defaults and provide additional data by providing a flat object
 * of paths and values.
 *
 * @function
 * @param {...Function} input Overrides
 * @returns {object}
 */
const mock = (input) => kit.expandWithPaths(path, {
  namespace: 'visibility',
  uuid: '423',
  ...input
});

/* ADD CUSTOM CONTENT BELOW */

/**
 * List of keys that are used in the public namespaces
 *
 * @enum {string}
 */
const publicKey = {
  /** Key used in session to mark a session as cleared before this timestamp */
  clearSessionTS: 'clearSessionTS'
};

/**
 * List of namespaces that are available to all plugins
 *
 * @enum {string}
 */
const publicNamespace = {
  /** Namespace for annotations that effect event visibility */
  visibility: 'visibility'
};

/**
 * Returns a path using the provided namespace. If the optional key is provided, it
 * will return that key from the payload.
 *
 * @example
 * import { kit, session, sessionAnnotation } from '@adobe/griffon-toolkit';
 * const publicPath = sessionAnnotation.makeNamespacePath('shared', 'pubic');
 *
 * const session = session.mock({
 *   annotations: [
 *     annotation.mock({
 *       [sessionAnnotation.path.namespace]: 'shared',
 *       [sessionAnnotation.path.payload]: { pubic: true }
 *     })
 *   ]
 * });
 *
 * console.log(kit.search(publicPath, session))); // [true]
 *
 * @function
 * @param {string} namespace The namespace to reference
 * @param {string} [key] Key inside payload
 * @returns {string}
 */
const makeNamespacePath = (namespace, key) => `(annotations[?namespace=='${namespace}'].payload${key ? `.${key}` : ''})[0]`;

/**
 * Returns a the value of the path using the provided namespace and key.
 *
 * @example
 * import { kit, session, sessionAnnotation } from '@adobe/griffon-toolkit';
 * const isPublic = sessionAnnotation.search('shared', 'pubic');
 *
 * const session = session.mock({
 *   annotations: [
 *     annotation.mock({
 *       [sessionAnnotation.path.namespace]: 'shared',
 *       [sessionAnnotation.path.payload]: { pubic: true }
 *     })
 *   ]
 * });
 *
 * console.log(isPublic(session)); // true
 *
 * @function
 * @param {string} namespace The namespace to reference
 * @param {string} [key] Key inside payload
 * @param {*} data Data to search
 * @returns {string}
 */
const search = kit.curry(
  (namespace, key, data) => kit.search(makeNamespacePath(namespace, key), data)
);

// additional exports should be added here:
const customExports = {
  publicKey, publicNamespace, makeNamespacePath, search
};

/* END CUSTOM CONTENT */
/* The content below is autogenerated. Do not make any changes */

export default {
  path,
  mock,
  make,
  schema,
  get,
  ...customExports,
  getPayload,
  getPayloadKey,
  getNamespace,
  getUuid,
  label,
  group,
  parentDepth
};