All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.myfaces.trinidadinternal.agent.parse.CapabilitiesDocument Maven / Gradle / Ivy

There is a newer version: 2.2.1
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * 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 CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.myfaces.trinidadinternal.agent.parse;

import org.apache.myfaces.trinidad.logging.TrinidadLogger;
import org.apache.myfaces.trinidad.context.Agent;

import java.util.ArrayList;
import java.util.List;
import java.util.HashMap;
import java.net.URL;


/**
 * implementation of Capabilities document
 * only the capabilities provider needs to know about the document.
 */
public class CapabilitiesDocument
{

  CapabilitiesDocument(CapabilitiesNode[] agents, DeviceNode[] devices)
  {
    _agents = agents;
    _devices = devices;
    _includeNodeBySrcCaps = new HashMap();
    _defaultAgentCapabilities = _getDefaultAgentCapabilities(_agents);
  }


  private CapabilitiesDocument()
  {
    this(null, null);
  }

  /**
   * @param agent
   * @return An array of Capabilities {name, value, name, value}.
   * The array may have duplicates and is ordered in ascending order (priority)
   */
  public Object[][] getCapabilities(Agent agent)
  {
    Object[][] agentCaps = null;
    //TODO: Right now we go by best match. But May be we should
    //get all that nodes that matched (sort by match score) and merge
    //This way the xml file would be simpler (--or more cryptic??)
    CapabilitiesNode agentMatchedNode = _getAgentCapabiltiesNode(agent);
    if (agentMatchedNode != null)
      agentCaps = _getCapabilities(agent, _agents, null, agentMatchedNode);



    //Now get the capabilities of the device, if it is known
    // - Match the correct device node
    // - Get the components in the device
    // - resolve the capabilities of the components
    // - If agent name is aavailable and is not "UNKNOWN" then
    //   use the capabilities of the agent (instead of the capabilities of the
    //   browser component). We need to do this as request could be from a agent
    //   like opera, but the device default browser may be something else.
    //TODO: Need to support a "extends" mechanism for devices, so that one device
    //can extend capabilities from a similar device.
    DeviceNode dNode = _getDeviceNode(agent);
    if (dNode != null)
    {
      DeviceComponentNode[] cNodes = dNode.__getComponents();
      if (cNodes == null)
        return agentCaps;

      Object[][][] deviceCaps = new Object[cNodes.length][][];
      for (int i = 0; i < cNodes.length; i++)
      {
        if ((cNodes[i].__getType() == DeviceComponentNode.TYPE_BROWSER) &&
            (agent.getAgentName() != null))
        {
          deviceCaps[i] = agentCaps;
        }
        else
        {
          deviceCaps[i] = _getCapabilities(agent, _agents, cNodes[i]);
        }
      }

      return _mergeCaps(deviceCaps, null);
    }

    //if device is null return agentCaps
    //if agentName is also null then return
    //default agentCaps
    if (agentCaps != null)
      return agentCaps;

    return _defaultAgentCapabilities;
  }


  private Object[][] _getDefaultAgentCapabilities(CapabilitiesNode[] agents)
  {
    if (_agents == null)
      return null;

    for (int i = 0; i < _agents.length; i++)
    {
      if (agents[i].__isDefault())
      {
        return _getCapabilities(null, _agents, null, _agents[i]);
      }
    }

    return null;
  }

  /**
   * @param agent
   * @return the capabilities node that best matches for current Agent
   */
  private CapabilitiesNode _getAgentCapabiltiesNode(Agent agent)
  {
    if (_agents == null)
      return null;
    //get the best match node
    double maxScore = 0;
    CapabilitiesNode agentMatchedNode = null;
    for (int i = 0; i < _agents.length; i++)
    {
      double score = _agents[i].__matches(agent);
      if ((score > 0) && (score > maxScore))
      {
        maxScore = score;
        agentMatchedNode = _agents[i];
      }
    }
    return agentMatchedNode;
  }


  /**
   * @param agent
   * @return device node that best matches the current Agent
   */
  private DeviceNode _getDeviceNode(Agent agent)
  {
    if (_devices == null)
       return null;

    double maxScore = 0;
    DeviceNode deviceNode = null;
    for (int i = 0; i < _devices.length; i++)
    {
      double score = _devices[i].__matches(agent);
      if ((score > 0) && (score > maxScore))
      {
        maxScore = score;
        deviceNode = _devices[i];
      }
    }
    return deviceNode;
  }


  /**
   * Get/resolve the capabilities for for a give capabilitiesNode
   */
  private Object[][] _getCapabilities(Agent agent,
                                      CapabilitiesNode[] capNodes,
                                      List includedByRefs,
                                      CapabilitiesNode matchNode)
  {
    assert (matchNode != null);

    if (capNodes == null)
      return new Object[0][0];

    //--do we need to bother about caching (a resolved state)
    //for each node? Not until this becomes a problem

    //check for circular dependecy
    if (includedByRefs == null)
      includedByRefs = new ArrayList();

    if (matchNode.__getId() != null)
    {
      if (includedByRefs.contains(matchNode.__getId()))
      {
        _LOG.warning("INVALID_DEPENDENCY");
        return new Object[0][0];
      }

      includedByRefs.add(matchNode.__getId());
    }

    //process node that includes by reference
    IncludeNode[] refIncludes = matchNode.__getIncludesByRef();
    Object[][][] refCaps = null;
    if (refIncludes != null)
    {
      refCaps = new Object[refIncludes.length][][];
      for (int i = 0; i < refIncludes.length; i++)
      {
        assert (refIncludes[i].__getRefId() != null);
        refCaps[i] = _getCapabilities(agent, capNodes,
                                      includedByRefs, refIncludes[i].__getRefId());
      }
    }

    //process nodes the includes a external file
    IncludeNode[] uriIncludes = matchNode.__getIncludesByUri();
    Object[][] uriCaps = null;
    if (uriIncludes != null)
    {
      uriCaps = new Object[uriIncludes.length][];
      for (int i = 0; i < uriIncludes.length; i++)
      {
        assert (uriIncludes[i].__getSrcUrl() != null);
        uriCaps[i] = _getCapabilities(uriIncludes[i].__getSrcUrl());
        assert (uriCaps[i] != null);
      }
    }

    return _mergeCaps(refCaps, uriCaps);
  }


  private Object[][] _getCapabilities(Agent agent,
                                      CapabilitiesNode[] capNodes,
                                      DeviceComponentNode dcNode)
  {

    assert (capNodes != null);
    assert (dcNode != null);

    ArrayList includedByRefs = new ArrayList();

    //process node that includes by reference
    IncludeNode[] refIncludes = dcNode.__getIncludesByRef();
    Object[][][] refCaps = null;
    if (refIncludes != null)
    {
      refCaps = new Object[refIncludes.length][][];
      for (int i = 0; i < refIncludes.length; i++)
      {
        assert (refIncludes[i].__getRefId() != null);
        refCaps[i] = _getCapabilities(agent, capNodes,
                                      includedByRefs, refIncludes[i].__getRefId());
      }
    }

    //process nodes the includes a external file
    IncludeNode[] uriIncludes = dcNode.__getIncludesByUri();
    Object[][] uriCaps = null;
    if (uriIncludes != null)
    {
      uriCaps = new Object[uriIncludes.length][];
      for (int i = 0; i < uriIncludes.length; i++)
      {
        assert (uriIncludes[i].__getSrcUrl() != null);
        uriCaps[i] = _getCapabilities(uriIncludes[i].__getSrcUrl());
        assert (uriCaps[i] != null);
      }
    }

    return _mergeCaps(refCaps, uriCaps);
  }


  /**
   * get capabilties of node using a node refid
   */
  private Object[][] _getCapabilities(Agent agent,
                                      CapabilitiesNode[] capNodes,
                                      List includedByRefs,
                                      String refId)
  {
    assert (capNodes != null);
    assert (refId != null);
    assert (includedByRefs != null);

    for (int i = 0; i < capNodes.length; i++)
    {
      if (refId.equals(capNodes[i].__getId()))
      {
        Object[][] caps = _getCapabilities(agent, capNodes, includedByRefs, capNodes[i]);
        return caps;
      }
    }

    _LOG.warning("REFERENCE_ID_NOT_FOUND", refId);
    return new Object[0][0];
  }


  /**
   * capabilities from parsed and cached capability data documents
   */
  private Object[] _getCapabilities(URL srcUrl)
  {
    Object o = _includeNodeBySrcCaps.get(srcUrl);
    if (o != null)
      return (Object[]) o;

    Object[] caps = null;
    synchronized (_includeNodeBySrcCaps)
    {
      caps = CapabilityDataDocumentParser.parse(srcUrl);
      _includeNodeBySrcCaps.put(srcUrl, caps);
    }

    return caps;
  }

  //merge capabilities array
  private Object[][] _mergeCaps(Object[][][] caps1, Object[][] caps2)
  {
    if (caps1 == null)
      return caps2;

    int len = 0;
    for (int i = 0; i < caps1.length; i++)
    {
      len += caps1[i].length;
    }

    Object[][] merged = new Object[len + (caps2 == null ? 0 : caps2.length)][];
    len = 0;
    for (int i = 0; i < caps1.length; i++)
    {
      System.arraycopy(caps1[i], 0, merged, len, caps1[i].length);
      len += caps1[i].length;
    }

    if (caps2 != null)
      System.arraycopy(caps2, 0, merged, len, caps2.length);

    return merged;
  }


  private CapabilitiesNode[] _agents;
  private DeviceNode[] _devices;
  private HashMap _includeNodeBySrcCaps;
  private Object[][] _defaultAgentCapabilities;

  static final CapabilitiesDocument EMPTY_DOCUMENT = new CapabilitiesDocument();

  private static TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger(CapabilitiesDocument.class);

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy