api_generator / branches / master / models / api_class.php

history
<?php
/**
 * Api Class Model
 *
 * Used for fetching information from the class index.
 *
 * PHP 5.2+
 *
 * CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
 * Copyright 2005-2009, Cake Software Foundation, Inc. (http://cakefoundation.org)
 *
 * Licensed under The MIT License
 * Redistributions of files must retain the above copyright notice.
 *
 * @copyright     Copyright 2005-2009, Cake Software Foundation, Inc. (http://cakefoundation.org)
 * @link          http://cakephp.org
 * @package       api_generator
 * @subpackage    api_generator.models
 * @since         ApiGenerator 0.1
 * @license       MIT License (http://www.opensource.org/licenses/mit-license.php)
 **/
class ApiClass extends ApiGeneratorAppModel {
/**
 * Name
 *
 * @var string
 */
	public $name = 'ApiClass';
/**
 * Validation rules
 *
 * @var string
 **/
	public $validate = array(
		'name' => array(
			'empty' => array(
				'rule' => 'notEmpty',
				'message' => 'Name must not be empty',
			)
		),
		'flags' => array(
			'number' => array(
				'rule' => 'numeric',
				'message' => 'Flags are numeric only',
			)
		)
	);
/**
 * belongsTo assoc
 *
 * @var array
 **/
	public $belongsTo = array(
		'ApiPackage' => array(
			'className' => 'ApiGenerator.ApiPackage',
			'foreign_key' => 'api_class_id',
		)
	);
/**
 * Flag bitmask for Pseudo classes (files with global functions)
 * get a pseudo class assigned to them
 *
 * @var int
 **/
	const PSEUDO_CLASS = 1;
/**
 * Concrete class bitmask;
 *
 * @var string
 **/
	const CONCRETE_CLASS = 2;
/**
 * Clears (truncates) the class index.
 *
 * @return void
 **/
	public function clearIndex() {
		$db = ConnectionManager::getDataSource($this->useDbConfig);
		$db->truncate($this->useTable);
	}
/**
 * save the entry in the index for a ClassDocumentor object
 *
 * @param object $classDoc Instance of ClassDocumentor to add to database.
 * @return boolean success
 **/
	public function saveClassDocs(ClassDocumentor $classDoc) {
		$classDoc->getAll();
		$slug = $this->_makeSlug($classDoc->name);
		$new = array(
			'name' => $classDoc->name,
			'slug' => $slug,
			'file_name' => $classDoc->classInfo['fileName'],
			'method_index' => $this->_generateIndex($classDoc, 'methods'),
			'property_index' => $this->_generateIndex($classDoc, 'properties'),
			'flags' => ApiClass::CONCRETE_CLASS,
		);
		$this->set($new);
		return $this->save();
	}
/**
 * Save a set of global functions to the ApiClass index.
 * Will make one record with a 'class name' derived from the filename.
 *
 * @param array $functions Array of FunctionDocumentor objects to index.
 * @param string $filename Name of file these things are found in.
 * @return boolean
 **/
	public function savePseudoClassDocs($functions, $filename) {
		$methodList = array();
		$name = basename($filename);
		$slug = $this->_makeSlug($name);
		foreach ($functions as $func) {
			if ($func instanceof FunctionDocumentor) {
				$methodList[] = $func->getName();
			}
		}
		$data = array(
			'name' => $name,
			'slug' => $slug,
			'file_name' => $filename,
			'method_index' => implode($methodList, ' '),
			'flags' => ApiClass::PSEUDO_CLASS,
		);
		$this->set($data);
		return $this->save();
	}
/**
 * Get the class index listing
 * 
 * @param boolean $includePseudoClass Whether you want to include 'pseudo' classes (no actual class)
 * @return array
 **/
	public function getClassIndex($includePseudoClass = false) {
		$conditions = array();
		if (!$includePseudoClass) {
			$conditions['ApiClass.flags'] = ApiClass::CONCRETE_CLASS;
		}
		return $this->find('list', array(
			'fields' => array('slug', 'name'), 
			'order' => 'ApiClass.name ASC',
			'conditions' => $conditions
		));
	}
/**
 * Generate a search index of methods or properties for the ClassDocumentor Object
 *
 * @param mixed $classDoc
 * @param string $what
 * @return void
 * @access protected
 */
	protected function _generateIndex(&$classDoc, $what = 'methods') {
		$index = array();
		foreach ((array)$classDoc->$what as $result) {
			if ($result['declaredInClass'] != $classDoc->classInfo['name']) {
				continue;
			}
			$index[] = $result['name'];
		}
		return strtolower(implode($index, ' '));
	}
/**
 * search method
 *
 * Find matching records for the given term or terms
 * Find results ordered by those matching in order: class names, method names, properties
 *
 * @param mixed $terms array of terms or search term
 * @return array of matching ApiFile objects
 * @access public
 */
	function search($terms = array()) {
		if (!$terms) {
			return array();
		}
		$terms = array_map('strtolower', (array)$terms);
		$fields = array('DISTINCT ApiClass.id', 'ApiClass.name', 'ApiClass.method_index',
			'ApiClass.property_index', 'file_name');
		$order = 'ApiClass.name';

		$conditions = array();
		foreach ($terms as $term) {
			$conditions['OR'][] = array('ApiClass.name LIKE' => '%' . $term . '%');
			$conditions['OR'][] = array('ApiClass.slug LIKE' => '%' . $term . '%');
			$conditions['OR'][] = array('ApiClass.method_index LIKE' => '%' . $term . '%');
			$conditions['OR'][] = array('ApiClass.property_index LIKE' => '%' . $term . '%');
		}
		$results = $this->find('all', compact('conditions', 'order', 'fields'));
		return $this->_queryFiles($results, $terms);
	}
/**
 * filterSearchResults method
 *
 * Purge results that don't match the search terms
 *
 * @param array $results
 * @param array $terms
 * @return array filtered results
 * @access protected
 */
	protected function _queryFiles($results, $terms) {
		if (!defined('DISABLE_AUTO_DISPATCH')) {
			define('DISABLE_AUTO_DISPATCH', true);
		}
		$return = $_return = array();
		$searchedClasses = Set::extract('/ApiClass/name', $results);

		$ApiFile =& ClassRegistry::init('ApiGenerator.ApiFile');
		foreach ($results as $i => $record) {
			$result = $ApiFile->loadFile($record['ApiClass']['file_name'], array('useIndex' => true));
			foreach ($result['class'] as $name => $obj) {
				if (!in_array($name, $searchedClasses)) {
					continue;
				}
				$relevance = 0;
				$this->_unsetUnmatching($obj, $terms, 'properties');
				$this->_unsetUnmatching($obj, $terms, 'methods');
				$relevance += $this->_calculateRelevance(array(compact('name')), $terms, array('high' => 6, 'low' => 3));
				if ($obj->methods) {
					$relevance += $this->_calculateRelevance($obj->methods, $terms);
				}
				if ($obj->properties) {
					$relevance += $this->_calculateRelevance($obj->properties, $terms);
				}
				$_return[$relevance][$name]['class'][$name] = $obj;
			}
			foreach ($result['function'] as $name => $obj) {
				$obj->info['declaredInFile'] = $record['ApiClass']['file_name'];
				$relevance = 0;
				$relevance += $this->_calculateRelevance(array(compact('name')), $terms, array('high' => 6, 'low' => 3));
				if ($relevance > 0) {
					$_return[$relevance][$name]['function'][$name] = $obj;
				}
			}
		}
		ksort($_return);
		$_return = array_reverse($_return);
		foreach ($_return as $result) {
			ksort($result);
			$return = am($return, $result);
		}
		return $return;
	}
/**
 * calculate the relevance of a match.
 *
 * @param array $subjects Things to calculate relevance for.
 * @param array $terms Terms that were searched for.
 * @param array $spread array of 'high' and 'low' relevance amounts
 * @return int
 **/
	protected function _calculateRelevance($subjects, $terms, $spread = array('high' => 4, 'low' => 2)) {
		$relevance = 0;
		foreach ($subjects as $subject) {
			$low = strtolower($subject['name']);
			foreach ($terms as $term) {
				if ($low === $term) {
					$relevance += $spread['high'];
				} elseif (strpos($low, $term) !== false) {
					$relevance += $spread['low'];
				}
			}
		}
		return $relevance;
	}
/**
 * unsetUnmatching method
 *
 * @param mixed $obj
 * @param array $terms
 * @param string $field
 * @return void
 * @access protected
 */
	function _unsetUnmatching(&$obj, $terms = array(), $field = 'properties') {
		if (empty($obj->$field)) {
			return;
		}
		foreach ($obj->$field as $j => $prop) {
			$delete = true;
			foreach($terms as $term) {
				if (strpos(strtolower($prop['name']), $term) !== false) {
					$delete = false;
					break;
				}
			}
			if ($delete) {
				unset ($obj->{$field}[$j]);
			}
		}
	}

/**
 * Analyzes Documentation coverage.
 * Use this method if you are unsure of the contents of an apiClass record, or
 * don't already have the reflection objects.
 * 
 * @param array $apiClass An ApiClass record to be analyzed.
 * @return array Array of warnings / info / % complete
 * @throws Exception  Throws exception if you are looking at a non-concrete class, of if there
 *  was an error with analyzation.
 **/
	public function analyzeCoverage($apiClass) {
		App::import('Vendor', 'ApiGenerator.DocBlockAnalyzer');
		$className = $apiClass['ApiClass']['name'];

		$ApiFile = ClassRegistry::init('ApiFile');
		$docsObjects = $ApiFile->loadFile($apiClass['ApiClass']['file_name'], array('useIndex' => true));
		if ($apiClass['ApiClass']['flags'] & ApiClass::CONCRETE_CLASS) {
			$Analyzer = new DocBlockAnalyzer();
			$Analyzer->setSource($docsObjects['class'][$className]);
			$coverage = $Analyzer->analyze();
			$this->id = $apiClass['ApiClass']['id'];
			$this->saveField('coverage_cache', $coverage['finalScore']);
			return $coverage;
		}
		throw new Exception(sprintf(
			__('%s is a pseudo class, and cannot have coverage generated', true),
			$apiClass['ApiClass']['name']
		));
	}
}
?>