<?php
// http://code.google.com/p/simple-linkedinphp/
// 3.2.0 - November 29, 2011
// hacked into the code to handel new scope (r_basicprofile+r_emailaddress) - until Paul update linkedinphp library!
// Facyla note 20131219 : this in fact should not be hacked, as Linkedin lets developpers define the wanted scope
//   in Linkedin application settings, when creating the (required) application and API access

/**
 * This file defines the 'LinkedIn' class. This class is designed to be a
 * simple, stand-alone implementation of the LinkedIn API functions.
 *
 * COPYRIGHT:
 *
 * Copyright (C) 2011, fiftyMission Inc.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
 * IN THE SOFTWARE.
 *
 * SOURCE CODE LOCATION:
 *
 * http://code.google.com/p/simple-linkedinphp/
 *
 * REQUIREMENTS:
 *
 * 1. You must have cURL installed on the server and available to PHP.
 * 2. You must be running PHP 5+.
 *
 * QUICK START:
 *
 * There are two files needed to enable LinkedIn API functionality from PHP; the
 * stand-alone OAuth library, and this LinkedIn class. The latest version of
 * the stand-alone OAuth library can be found on Google Code:
 *
 * http://code.google.com/p/oauth/
 *
 * Install these two files on your server in a location that is accessible to
 * the scripts you wish to use them in. Make sure to change the file
 * permissions such that your web server can read the files.
 *
 * Next, make sure the path to the OAuth library is correct (you can change this
 * as needed, depending on your file organization scheme, etc).
 *
 * Finally, test the class by attempting to connect to LinkedIn using the
 * associated demo.php page, also located at the Google Code location
 * referenced above.
 *
 * RESOURCES:
 *
 * REST API Documentation: http://developer.linkedin.com/rest
 *
 * @version 3.2.0 - November 8, 2011
 * @author Paul Mennega <paul@fiftymission.net>
 * @copyright Copyright 2011, fiftyMission Inc.
 * @license http://www.opensource.org/licenses/mit-license.php The MIT License
 */

/**
 * 'LinkedInException' class declaration.
 *
 * This class extends the base 'Exception' class.
 *
 * @access public
 * @package classpackage
 */
class LinkedInException extends Exception {}

/**
 * 'LinkedIn' class declaration.
 *
 * This class provides generalized LinkedIn oauth functionality.
 *
 * @access public
 * @package classpackage
 */
class LinkedIn {
  // api/oauth settings
  const _API_OAUTH_REALM             = 'http://api.linkedin.com';
  const _API_OAUTH_VERSION           = '1.0';

  // the default response format from LinkedIn
  const _DEFAULT_RESPONSE_FORMAT     = 'xml';

  // helper constants used to standardize LinkedIn <-> API communication.  See demo page for usage.
  const _GET_RESPONSE                = 'lResponse';
  const _GET_TYPE                    = 'lType';

  // Invitation API constants.
  const _INV_SUBJECT                 = 'Invitation to connect';
  const _INV_BODY_LENGTH             = 200;

  // API methods
  const _METHOD_TOKENS               = 'POST';

  // Network API constants.
  const _NETWORK_LENGTH              = 1000;
  const _NETWORK_HTML                = '<a>';

  // response format type constants, see http://developer.linkedin.com/docs/DOC-1203
  const _RESPONSE_JSON               = 'JSON';
  const _RESPONSE_JSONP              = 'JSONP';
  const _RESPONSE_XML                = 'XML';

  // Share API constants
  const _SHARE_COMMENT_LENGTH        = 700;
  const _SHARE_CONTENT_TITLE_LENGTH  = 200;
  const _SHARE_CONTENT_DESC_LENGTH   = 400;

  // LinkedIn API end-points
	const _URL_ACCESS                  = 'https://api.linkedin.com/uas/oauth/accessToken';
	const _URL_API                     = 'https://api.linkedin.com';
	const _URL_AUTH                    = 'https://www.linkedin.com/uas/oauth/authenticate?oauth_token=';
	const _URL_REQUEST                 = 'https://api.linkedin.com/uas/oauth/requestToken';
	// const _URL_REQUEST                 = 'https://api.linkedin.com/uas/oauth/requestToken?scope=r_basicprofile+r_emailaddress+rw_nus+r_network';
	const _URL_REVOKE                  = 'https://api.linkedin.com/uas/oauth/invalidateToken';

	// Library version
	const _VERSION                     = '3.2.0';

  // oauth properties
  protected $callback;
  protected $token                   = null;

  // application properties
  protected $application_key,
            $application_secret;

  // the format of the data to return
  protected $response_format         = self::_DEFAULT_RESPONSE_FORMAT;

  // last request fields
  public $last_request_headers,
         $last_request_url;

	/**
	 * Create a LinkedIn object, used for OAuth-based authentication and
	 * communication with the LinkedIn API.
	 *
	 * @param arr $config
	 *    The 'start-up' object properties:
	 *           - appKey       => The application's API key
	 *           - appSecret    => The application's secret key
	 *           - callbackUrl  => [OPTIONAL] the callback URL
	 *
	 * @return obj
	 *    A new LinkedIn object.
	 */
	public function __construct($config) {
    if(!is_array($config)) {
      // bad data passed
		  throw new LinkedInException('LinkedIn->__construct(): bad data passed, $config must be of type array.');
    }
    $this->setApplicationKey($config['appKey']);
	  $this->setApplicationSecret($config['appSecret']);
	  $this->setCallbackUrl($config['callbackUrl']);
	}

	/**
   * The class destructor.
   *
   * Explicitly clears LinkedIn object from memory upon destruction.
	 */
  public function __destruct() {
    unset($this);
	}

	/**
	 * Bookmark a job.
	 *
	 * Calling this method causes the current user to add a bookmark for the
	 * specified job:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1323
	 *
	 * @param str $jid
	 *    Job ID you want to bookmark.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function bookmarkJob($jid) {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->bookmarkJob(): bad data passed, $jid must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/job-bookmarks';
	  $response = $this->fetch('POST', $query, '<job-bookmark><job><id>' . trim($jid) . '</id></job></job-bookmark>');

	  /**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(201, $response);
	}

	/**
	 * Get list of jobs you have bookmarked.
	 *
	 * Returns a list of jobs the current user has bookmarked, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1323
	 *
	 * @return arr
	 *         array containing retrieval success, LinkedIn response.
	 */
	public function bookmarkedJobs() {
    // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/job-bookmarks';
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * Custom addition to make code compatible with PHP 5.2
	 */
	private function intWalker($value, $key) {
        if(!is_int($value)) {
			throw new LinkedInException('LinkedIn->checkResponse(): $http_code_required must be an integer or an array of integer values');
		}
    }

	/**
	 * Used to check whether a response LinkedIn object has the required http_code or not and
	 * returns an appropriate LinkedIn object.
	 *
	 * @param var $http_code_required
	 * 		The required http response from LinkedIn, passed in either as an integer,
	 * 		or an array of integers representing the expected values.
	 * @param arr $response
	 *    An array containing a LinkedIn response.
	 *
	 * @return boolean
	 * 	  true or false depending on if the passed LinkedIn response matches the expected response.
	 */
	private function checkResponse($http_code_required, $response) {
		// check passed data
    if(is_array($http_code_required)) {
		  array_walk($http_code_required, array($this, 'intWalker'));
		} else {
		  if(!is_int($http_code_required)) {
  			throw new LinkedInException('LinkedIn->checkResponse(): $http_code_required must be an integer or an array of integer values');
  		} else {
  		  $http_code_required = array($http_code_required);
  		}
		}
		if(!is_array($response)) {
			throw new LinkedInException('LinkedIn->checkResponse(): $response must be an array');
		}

		// check for a match
		if(in_array($response['info']['http_code'], $http_code_required)) {
		  // response found
		  $response['success'] = true;
		} else {
			// response not found
			$response['success'] = false;
			$response['error']   = 'HTTP response from LinkedIn end-point was not code ' . implode(', ', $http_code_required);
		}
		return $response;
	}

	/**
	 * Close a job.
	 *
	 * Calling this method causes the passed job to be closed, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1151
	 *
	 * @param str $jid
	 *    Job ID you want to close.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function closeJob($jid) {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->closeJob(): bad data passed, $jid must be of string value.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/jobs/partner-job-id=' . trim($jid);
	  $response = $this->fetch('DELETE', $query);

	  /**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(204, $response);
	}

	/**
	 * Share comment posting method.
	 *
	 * Post a comment on an existing connections shared content. API details can
	 * be found here:
	 *
	 * http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $uid
	 *    The LinkedIn update ID.
	 * @param str $comment
	 *    The share comment to be posted.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function comment($uid, $comment) {
	  // check passed data
	  if(!is_string($uid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->comment(): bad data passed, $uid must be of type string.');
	  }
    if(!is_string($comment)) {
      // nothing/non-string passed, raise an exception
		  throw new LinkedInException('LinkedIn->comment(): bad data passed, $comment must be a non-zero length string.');
    }

    /**
     * Share comment rules:
     *
     * 1) No HTML permitted.
     * 2) Comment cannot be longer than 700 characters.
     */
    $comment = substr(trim(htmlspecialchars(strip_tags($comment))), 0, self::_SHARE_COMMENT_LENGTH);
		$data    = '<?xml version="1.0" encoding="UTF-8"?>
                <update-comment>
  				        <comment>' . $comment . '</comment>
  				      </update-comment>';

    // construct and send the request
    $query    = self::_URL_API . '/v1/people/~/network/updates/key=' . $uid . '/update-comments';
    $response = $this->fetch('POST', $query, $data);

    /**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * Share comment retrieval.
	 *
	 * Return all comments associated with a given network update:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $uid
	 *    The LinkedIn update ID.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function comments($uid) {
	  // check passed data
	  if(!is_string($uid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->comments(): bad data passed, $uid must be of type string.');
	  }

		// construct and send the request
    $query    = self::_URL_API . '/v1/people/~/network/updates/key=' . $uid . '/update-comments';
    $response = $this->fetch('GET', $query);

  	/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(200, $response);
	}

	/**
	 * Company profile retrieval function.
	 *
	 * Takes a string of parameters as input and requests company profile data
	 * from the LinkedIn Company Profile API. See the official documentation for
	 * $options 'field selector' formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1014
	 *   http://developer.linkedin.com/docs/DOC-1259
	 *
	 * @param str $options
	 *    Data retrieval options.
	 * @param	bool $by_email
	 *    [OPTIONAL] Search by email domain?
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function company($options, $by_email = false) {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->company(): bad data passed, $options must be of type string.');
	  }
	  if(!is_bool($by_email)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->company(): bad data passed, $by_email must be of type boolean.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/companies' . ($by_email ? '' : '/') . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

  /**
	 * Company products and their associated recommendations.
	 *
	 * The product data type contains details about a company's product or
	 * service, including recommendations from LinkedIn members, and replies from
	 * company representatives.
	 *
	 *   http://developer.linkedin.com/docs/DOC-1327
	 *
	 * @param str $cid
	 *    Company ID you want the product for.
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function companyProducts($cid, $options = '') {
	  // check passed data
	  if(!is_string($cid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->companyProducts(): bad data passed, $cid must be of type string.');
	  }
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->companyProducts(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/companies/' . trim($cid) . '/products' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Connection retrieval function.
	 *
	 * Takes a string of parameters as input and requests connection-related data
	 * from the Linkedin Connections API. See the official documentation for
	 * $options 'field selector' formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1014
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function connections($options = '~/connections') {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->connections(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * This creates a post in the specified group with the specified title and specified summary.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 * @param str $title
	 * 		The title of the post. This must be non-empty.
	 * @param str $summary
	 * 		[OPTIONAL] The content or summary of the post. This can be empty.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function createPost($gid, $title, $summary = '') {
		if(!is_string($gid)) {
			throw new LinkedInException('LinkedIn->createPost(): bad data passed, $gid must be of type string.');
		}
		if(!is_string($title) || empty($title)) {
			throw new LinkedInException('LinkedIn->createPost(): bad data passed, $title must be a non-empty string.');
		}
		if(!is_string($summary)) {
			throw new LinkedInException('LinkedIn->createPost(): bad data passed, $summary must be of type string.');
		}

		// construct the XML
		$data = '<?xml version="1.0" encoding="UTF-8"?>
    				 <post>
    					 <title>'. $title . '</title>
    					 <summary>' . $summary . '</summary>
    				 </post>';

 		// construct and send the request
		$query    = self::_URL_API . '/v1/groups/' . trim($gid) . '/posts';
		$response = $this->fetch('POST', $query, $data);

	  /**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(201, $response);
	}

	/**
	 * This deletes the specified post if you are the owner or moderator that post.
	 * Otherwise, it just flags the post as inappropriate.
	 *
	 * https://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function deletePost($pid) {
		if(!is_string($pid)) {
			throw new LinkedInException('LinkedIn->deletePost(): bad data passed, $pid must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . trim($pid);
		$response = $this->fetch('DELETE', $query);

    /**
     * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Edit a job.
	 *
	 * Calling this method causes the passed job to be edited, with the passed
	 * XML instructing which fields to change, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1154
	 *   http://developer.linkedin.com/docs/DOC-1142
	 *
	 * @param str $jid
	 *    Job ID you want to renew.
	 * @param str $xml
	 *    The XML containing the job fields to edit.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function editJob($jid, $xml) {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->editJob(): bad data passed, $jid must be of string value.');
	  }
	  if(is_string($xml)) {
	    $xml = trim(stripslashes($xml));
	  } else {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->editJob(): bad data passed, $xml must be of string value.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/jobs/partner-job-id=' . trim($jid);
	  $response = $this->fetch('PUT', $query, $xml);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * General data send/request method.
	 *
	 * @param str $method
	 *    The data communication method.
	 * @param str $url
	 *    The Linkedin API endpoint to connect with.
	 * @param str $data
	 *    [OPTIONAL] The data to send to LinkedIn.
	 * @param arr $parameters
	 *    [OPTIONAL] Addition OAuth parameters to send to LinkedIn.
	 *
	 * @return arr
	 *    array containing:
	 *
	 *           array(
	 *             'info'      =>	Connection information,
	 *             'linkedin'  => LinkedIn response,
	 *             'oauth'     => The OAuth request string that was sent to LinkedIn
	 *           )
	 */
	protected function fetch($method, $url, $data = null, $parameters = array()) {
	  // check for cURL
	  if(!extension_loaded('curl')) {
	    // cURL not present
      throw new LinkedInException('LinkedIn->fetch(): PHP cURL extension does not appear to be loaded/present.');
	  }

    try {
	    // generate OAuth values
	    $oauth_consumer  = new OAuthConsumer($this->getApplicationKey(), $this->getApplicationSecret(), $this->getCallbackUrl());
	    $oauth_token     = $this->getToken();
	    $oauth_token     = (!is_null($oauth_token)) ? new OAuthToken($oauth_token['oauth_token'], $oauth_token['oauth_token_secret']) : null;
      $defaults        = array(
        'oauth_version' => self::_API_OAUTH_VERSION
      );
	    $parameters    = array_merge($defaults, $parameters);

	    // generate OAuth request
  		$oauth_req = OAuthRequest::from_consumer_and_token($oauth_consumer, $oauth_token, $method, $url, $parameters);
      $oauth_req->sign_request(new OAuthSignatureMethod_HMAC_SHA1(), $oauth_consumer, $oauth_token);

      // start cURL, checking for a successful initiation
      if(!$handle = curl_init()) {
         // cURL failed to start
        throw new LinkedInException('LinkedIn->fetch(): cURL did not initialize properly.');
      }

      // set cURL options, based on parameters passed
	    curl_setopt($handle, CURLOPT_CUSTOMREQUEST, $method);
      curl_setopt($handle, CURLOPT_RETURNTRANSFER, true);
      curl_setopt($handle, CURLOPT_SSL_VERIFYPEER, false);
      curl_setopt($handle, CURLOPT_URL, $url);
      curl_setopt($handle, CURLOPT_VERBOSE, false);

      // Restrict the request timeout to 5 seconds. Linkedin is sometimes very
      // slow and we don't want to trigger a PHP timeout on our end.
      curl_setopt($handle, CURLOPT_TIMEOUT, 5);

      if ( isset ( Hybrid_Auth::$config["proxy"] ) ) {
      	curl_setopt($handle, CURLOPT_PROXY, Hybrid_Auth::$config["proxy"]);
      }

      // configure the header we are sending to LinkedIn - http://developer.linkedin.com/docs/DOC-1203
      $header = array($oauth_req->to_header(self::_API_OAUTH_REALM));
      if(is_null($data)) {
        // not sending data, identify the content type
        $header[] = 'Content-Type: text/plain; charset=UTF-8';
        switch($this->getResponseFormat()) {
          case self::_RESPONSE_JSON:
            $header[] = 'x-li-format: json';
            break;
          case self::_RESPONSE_JSONP:
            $header[] = 'x-li-format: jsonp';
            break;
        }
      } else {
        $header[] = 'Content-Type: text/xml; charset=UTF-8';
        curl_setopt($handle, CURLOPT_POSTFIELDS, $data);
      }
      curl_setopt($handle, CURLOPT_HTTPHEADER, $header);

      // set the last url, headers
      $this->last_request_url = $url;
      $this->last_request_headers = $header;

      // gather the response
      $return_data['linkedin']        = curl_exec($handle);
      if( $return_data['linkedin'] === false ) {
          Hybrid_Logger::error( "LinkedIn::fetch(). curl_exec error: ", curl_error($handle) );
      }
      $return_data['info']            = curl_getinfo($handle);
      $return_data['oauth']['header'] = $oauth_req->to_header(self::_API_OAUTH_REALM);
      $return_data['oauth']['string'] = $oauth_req->base_string;

      // check for throttling
      if(self::isThrottled($return_data['linkedin'])) {
        throw new LinkedInException('LinkedIn->fetch(): throttling limit for this user/application has been reached for LinkedIn resource - ' . $url);
      }

      //TODO - add check for NO response (http_code = 0) from cURL

      // close cURL connection
      curl_close($handle);

      // no exceptions thrown, return the data
      return $return_data;
    } catch(OAuthException $e) {
      // oauth exception raised
      throw new LinkedInException('OAuth exception caught: ' . $e->getMessage());
    }
	}

	/**
	 * This flags a specified post as specified by type.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 * @param str $type
	 * 		The type to flag the post as.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function flagPost($pid, $type) {
		if(!is_string($pid)) {
			throw new LinkedInException('LinkedIn->flagPost(): bad data passed, $pid must be of type string');
		}
		if(!is_string($type)) {
			throw new LinkedInException('LinkedIn->flagPost(): bad data passed, $like must be of type string');
		}
		//Constructing the xml
		$data = '<?xml version="1.0" encoding="UTF-8"?>';
		switch($type) {
			case 'promotion':
				$data .= '<code>promotion</code>';
				break;
			case 'job':
				$data .= '<code>job</code>';
				break;
			default:
				throw new LinkedInException('LinkedIn->flagPost(): invalid value for $type, must be one of: "promotion", "job"');
				break;
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . $pid . '/category/code';
		$response = $this->fetch('PUT', $query, $data);

  	/**
     * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Follow a company.
	 *
	 * Calling this method causes the current user to start following the
	 * specified company, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1324
	 *
	 * @param str $cid
	 *    Company ID you want to follow.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function followCompany($cid) {
	  // check passed data
	  if(!is_string($cid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->followCompany(): bad data passed, $cid must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/following/companies';
	  $response = $this->fetch('POST', $query, '<company><id>' . trim($cid) . '</id></company>');

	  /**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(201, $response);
	}

	/**
	 * Follows/Unfollows the specified post.
	 *
	 * https://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 * @param bool $follow
	 * 		Determines whether to follow or unfollow the post. true = follow, false = unfollow
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */

	public function followPost($pid, $follow) {
		if(!is_string($pid)) {
			throw new LinkedInException('LinkedIn->followPost(): bad data passed, $pid must be of type string');
		}
		if(!($follow === true || $follow === false)) {
			throw new LinkedInException('LinkedIn->followPost(): bad data passed, $follow must be of type boolean');
		}

		// construct the XML
		$data = '<?xml version="1.0" encoding="UTF-8"?>
				     <is-following>'. (($follow) ? 'true' : 'false'). '</is-following>';

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . trim($pid) . '/relation-to-viewer/is-following';
		$response = $this->fetch('PUT', $query, $data);

		/**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Get list of companies you follow.
	 *
	 * Returns a list of companies the current user is currently following, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1324
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function followedCompanies() {
	  // construct and send the request
    $query    = self::_URL_API . '/v1/people/~/following/companies';
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Get the application_key property.
	 *
	 * @return str
	 *    The application key.
	 */
	public function getApplicationKey() {
	  return $this->application_key;
	}

	/**
	 * Get the application_secret property.
	 *
	 * @return str
	 *    The application secret.
	 */
	public function getApplicationSecret() {
	  return $this->application_secret;
	}

	/**
	 * Get the callback property.
	 *
	 * @return str
	 *    The callback url.
	 */
	public function getCallbackUrl() {
	  return $this->callback;
	}

  /**
	 * Get the response_format property.
	 *
	 * @return str
	 *    The response format.
	 */
	public function getResponseFormat() {
	  return $this->response_format;
	}

	/**
	 * Get the token_access property.
	 *
	 * @return arr
	 *    The access token.
	 */
	public function getToken() {
	  return $this->token;
	}

	/**
	 * [DEPRECATED] Get the token_access property.
	 *
	 * @return arr
	 *    The access token.
	 */
	public function getTokenAccess() {
	  return $this->getToken();
	}

	/**
	 *
	 * Get information about a specific group.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 	 	The group id.
	 *
	 * @param str $options
	 * 		[OPTIONAL] Field selectors for the group.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */

	public function group($gid, $options = '') {
		if(!is_string($gid)){
			throw new LinkedInException('LinkedIn->group(): bad data passed, $gid must be of type string.');
		}
		if(!is_string($options)) {
			throw new LinkedInException('LinkedIn->group(): bad data passed, $options must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/groups/' . trim($gid) . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * This returns all the groups the user is a member of.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $options
	 * 		[OPTIONAL] Field selectors for the groups.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function groupMemberships($options = '') {
		if(!is_string($options)) {
			throw new LinkedInException('LinkedIn->groupMemberships(): bad data passed, $options must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/group-memberships' . trim($options) . '?membership-state=member';
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * This gets a specified post made within a group.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 * @param str $options
	 * 		[OPTIONAL] Field selectors for the post.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function groupPost($pid, $options = '') {
		if(!is_string($pid)) {
			throw new LinkedInException('LinkedIn->groupPost(): bad data passed, $pid must be of type string.');
		}
		if(!is_string($options)) {
			throw new LinkedInException('LinkedIn->groupPost(): bad data passed, $options must be of type string.');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . trim($pid) . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * This returns all the comments made on the specified post within a group.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 * @param str $options
	 * 		[OPTIONAL] Field selectors for the post comments.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function groupPostComments($pid, $options = ''){
		if(!is_string($pid)){
			throw new LinkedInException('LinkedIn->groupPostComments(): bad data passed, $pid must be of type string.');
		}
		if(!is_string($options)) {
			throw new LinkedInException('LinkedIn->groupPostComments(): bad data passed, $options must be of type string.');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . trim($pid) . '/comments' . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * This returns all the posts within a group.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function groupPosts($gid, $options = '') {
		if(!is_string($gid)){
			throw new LinkedInException('LinkedIn->groupPosts(): bad data passed, $gid must be of type string');
		}
		if(!is_string($options)){
			throw new LinkedInException('LinkedIn->groupPosts(): bad data passed, $options must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/groups/' . trim($gid)  .'/posts' . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * This returns the group settings of the specified group
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 * @param str $options
	 * 		[OPTIONAL] Field selectors for the group.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function groupSettings($gid, $options = '') {
		if(!is_string($gid)) {
			throw new LinkedInException('LinkedIn->groupSettings(): bad data passed, $gid must be of type string');
		}
		if(!is_string($options)) {
			throw new LinkedInException('LinkedIn->groupSettings(): bad data passed, $options must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/group-memberships/' . trim($gid) . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * Send connection invitations.
	 *
	 * Send an invitation to connect to your network, either by email address or
	 * by LinkedIn ID. Details on the API here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1012
	 *
	 * @param str $method
	 *    The invitation method to process.
	 * @param str $recipient
	 *    The email/id to send the invitation to.
	 * @param str $subject
	 *    The subject of the invitation to send.
	 * @param str $body
	 *    The body of the invitation to send.
	 * @param str $type
	 *    [OPTIONAL] The invitation request type (only friend is supported at this time by the Invite API).
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function invite($method, $recipient, $subject, $body, $type = 'friend') {
    /**
     * Clean up the passed data per these rules:
     *
     * 1) Message must be sent to one recipient (only a single recipient permitted for the Invitation API)
     * 2) No HTML permitted
     * 3) 200 characters max in the invitation subject
     * 4) Only able to connect as a friend at this point
     */
    // check passed data
    if(empty($recipient)) {
   		throw new LinkedInException('LinkedIn->invite(): you must provide an invitation recipient.');
    }
    switch($method) {
      case 'email':
        if(is_array($recipient)) {
          $recipient = array_map('trim', $recipient);
        } else {
          // bad format for recipient for email method
          throw new LinkedInException('LinkedIn->invite(): invitation recipient email/name array is malformed.');
        }
        break;
      case 'id':
        $recipient = trim($recipient);
        if(!self::isId($recipient)) {
          // bad format for recipient for id method
          throw new LinkedInException('LinkedIn->invite(): invitation recipient ID does not match LinkedIn format.');
        }
        break;
      default:
        throw new LinkedInException('LinkedIn->invite(): bad invitation method, must be one of: email, id.');
        break;
    }
    if(!empty($subject)) {
      $subject = trim(htmlspecialchars(strip_tags(stripslashes($subject))));
    } else {
      throw new LinkedInException('LinkedIn->invite(): message subject is empty.');
    }
    if(!empty($body)) {
      $body = trim(htmlspecialchars(strip_tags(stripslashes($body))));
      if(strlen($body) > self::_INV_BODY_LENGTH) {
        throw new LinkedInException('LinkedIn->invite(): message body length is too long - max length is ' . self::_INV_BODY_LENGTH . ' characters.');
      }
    } else {
      throw new LinkedInException('LinkedIn->invite(): message body is empty.');
    }
    switch($type) {
      case 'friend':
        break;
      default:
        throw new LinkedInException('LinkedIn->invite(): bad invitation type, must be one of: friend.');
        break;
    }

    // construct the xml data
		$data   = '<?xml version="1.0" encoding="UTF-8"?>
		           <mailbox-item>
		             <recipients>
                   <recipient>';
                     switch($method) {
                       case 'email':
                         // email-based invitation
                         $data .= '<person path="/people/email=' . $recipient['email'] . '">
                                     <first-name>' . htmlspecialchars($recipient['first-name']) . '</first-name>
                                     <last-name>' . htmlspecialchars($recipient['last-name']) . '</last-name>
                                   </person>';
                         break;
                       case 'id':
                         // id-based invitation
                         $data .= '<person path="/people/id=' . $recipient . '"/>';
                         break;
                     }
    $data  .= '    </recipient>
                 </recipients>
                 <subject>' . $subject . '</subject>
                 <body>' . $body . '</body>
                 <item-content>
                   <invitation-request>
                     <connect-type>';
                       switch($type) {
                         case 'friend':
                           $data .= 'friend';
                           break;
                       }
    $data  .= '      </connect-type>';
                     switch($method) {
                       case 'id':
                         // id-based invitation, we need to get the authorization information
                         $query                 = 'id=' . $recipient . ':(api-standard-profile-request)';
                         $response              = self::profile($query);
                         if($response['info']['http_code'] == 200) {
                           $response['linkedin'] = self::xmlToarray($response['linkedin']);
                           if($response['linkedin'] === false) {
                             // bad XML data
                             throw new LinkedInException('LinkedIn->invite(): LinkedIn returned bad XML data.');
                           }
                           $authentication = explode(':', $response['linkedin']['person']['children']['api-standard-profile-request']['children']['headers']['children']['http-header']['children']['value']['content']);

                           // complete the xml
                           $data .= '<authorization>
                                       <name>' . $authentication[0] . '</name>
                                       <value>' . $authentication[1] . '</value>
                                     </authorization>';
                         } else {
                           // bad response from the profile request, not a valid ID?
                           throw new LinkedInException('LinkedIn->invite(): could not send invitation, LinkedIn says: ' . print_r($response['linkedin'], true));
                         }
                         break;
                     }
    $data  .= '    </invitation-request>
                 </item-content>
               </mailbox-item>';

    // send request
    $query    = self::_URL_API . '/v1/people/~/mailbox';
    $response = $this->fetch('POST', $query, $data);

		/**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * LinkedIn ID validation.
	 *
	 * Checks the passed string $id to see if it has a valid LinkedIn ID format,
	 * which is, as of October 15th, 2010:
	 *
	 *   10 alpha-numeric mixed-case characters, plus underscores and dashes.
	 *
	 * @param str $id
	 *    A possible LinkedIn ID.
	 *
	 * @return bool
	 *    true/false depending on valid ID format determination.
	 */
	public static function isId($id) {
	  // check passed data
    if(!is_string($id)) {
	    // bad data passed
	    throw new LinkedInException('LinkedIn->isId(): bad data passed, $id must be of type string.');
	  }

	  $pattern = '/^[a-z0-9_\-]{10}$/i';
	  if($match = preg_match($pattern, $id)) {
	    // we have a match
	    $return_data = true;
	  } else {
	    // no match
	    $return_data = false;
	  }
	  return $return_data;
	}

	/**
	 * Throttling check.
	 *
	 * Checks the passed LinkedIn response to see if we have hit a throttling
	 * limit:
	 *
	 * http://developer.linkedin.com/docs/DOC-1112
	 *
	 * @param arr $response
	 *    The LinkedIn response.
	 *
	 * @return bool
	 *    true/false depending on content of response.
	 */
	public static function isThrottled($response) {
	  $return_data = false;

    // check the variable
	  if(!empty($response) && is_string($response)) {
	    // we have an array and have a properly formatted LinkedIn response

      // store the response in a temp variable
      $temp_response = self::xmlToarray($response);
  	  if($temp_response !== false) {
    	  // check to see if we have an error
    	  if(array_key_exists('error', $temp_response) && ($temp_response['error']['children']['status']['content'] == 403) && preg_match('/throttle/i', $temp_response['error']['children']['message']['content'])) {
    	    // we have an error, it is 403 and we have hit a throttle limit
  	      $return_data = true;
    	  }
  	  }
  	}
  	return $return_data;
	}

	/**
	 * Job posting detail info retrieval function.
	 *
	 * The Jobs API returns detailed information about job postings on LinkedIn.
	 * Find the job summary, description, location, and apply our professional graph
	 * to present the relationship between the current member and the job poster or
	 * hiring manager.
	 *
	 *   http://developer.linkedin.com/docs/DOC-1322
	 *
	 * @param	str $jid
	 *    ID of the job you want to look up.
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function job($jid, $options = '') {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->job(): bad data passed, $jid must be of type string.');
	  }
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->job(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/jobs/' . trim($jid) . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Join the specified group, per:
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function joinGroup($gid) {
		if(!is_string($gid)) {
			throw new LinkedInException('LinkedIn->joinGroup(): bad data passed, $gid must be of type string.');
		}

		// constructing the XML
		$data = '<?xml version="1.0" encoding="UTF-8"?>
  				   <group-membership>
  				   	 <membership-state>
  				  	 	 <code>member</code>
  				  	 </membership-state>
  				   </group-membership>';

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/group-memberships/' . trim($gid);
		$response = $this->fetch('PUT', $query, $data);

		/**
	   * Check for successful request (a 200 or 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(array(200, 201), $response);
	}

	/**
	 * Returns the last request header from the previous call to the
	 * LinkedIn API.
	 *
	 * @returns str
	 *    The header, in string format.
	 */
	public function lastRequestHeader() {
	   return $this->last_request_headers;
	}

	/**
	 * Returns the last request url from the previous call to the
	 * LinkedIn API.
	 *
	 * @returns str
	 *    The url, in string format.
	 */
	public function lastRequestUrl() {
	   return $this->last_request_url;
	}

	/**
	 * Leave the specified group, per:.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function leaveGroup($gid){
		if(!is_string($gid)) {
			throw new LinkedInException('LinkedIn->leaveGroup(): bad data passed, $gid must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/group-memberships/'  .trim($gid);
		$response = $this->fetch('DELETE', $query);

		/**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Like another user's network update, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $uid
	 *    The LinkedIn update ID.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function like($uid) {
	  // check passed data
	  if(!is_string($uid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->like(): bad data passed, $uid must be of type string.');
	  }

    // construct the XML
		$data = '<?xml version="1.0" encoding="UTF-8"?>
		         <is-liked>true</is-liked>';

		// construct and send the request
    $query    = self::_URL_API . '/v1/people/~/network/updates/key=' . $uid . '/is-liked';
    $response = $this->fetch('PUT', $query, $data);

  	/**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * Likes/unlikes the specified post, per:
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $pid
	 * 		The post id.
	 * @param bool $like
	 * 		Determines whether to like or unlike. true = like, false = unlike.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function likePost($pid, $like) {
		if(!is_string($pid)) {
			throw new LinkedInException ('LinkedIn->likePost(): bad data passed, $pid must be of type string');
		}
		if(!($like === true || $like === false)) {
			throw new LinkedInException('LinkedIn->likePost(): bad data passed, $like must be of type boolean');
		}

		// construct the XML
		$data = '<?xml version="1.0" encoding="UTF-8"?>
		         <is-liked>'.(($like) ? 'true': 'false').'</is-liked>';

		// construct and send the request
		$query    = self::_URL_API . '/v1/posts/' . trim($pid) . '/relation-to-viewer/is-liked';
		$response = $this->fetch('PUT', $query, $data);

		/**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Retrieve network update likes.
	 *
	 * Return all likes associated with a given network update:
	 *
	 * http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $uid
	 *    The LinkedIn update ID.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function likes($uid) {
	  // check passed data
	  if(!is_string($uid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->likes(): bad data passed, $uid must be of type string.');
	  }

		// construct and send the request
    $query    = self::_URL_API . '/v1/people/~/network/updates/key=' . $uid . '/likes';
    $response = $this->fetch('GET', $query);

  	/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(200, $response);
	}

	/**
	 * Connection messaging method.
	 *
	 * Send a message to your network connection(s), optionally copying yourself.
	 * Full details from LinkedIn on this functionality can be found here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1044
	 *
	 * @param arr $recipients
	 *    The connection(s) to send the message to.
	 * @param str $subject
	 *    The subject of the message to send.
	 * @param str $body
	 *    The body of the message to send.
	 * @param bool $copy_self
	 *    [OPTIONAL] Also update the teathered Twitter account.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function message($recipients, $subject, $body, $copy_self = false) {
    /**
     * Clean up the passed data per these rules:
     *
     * 1) Message must be sent to at least one recipient
     * 2) No HTML permitted
     */
    if(!empty($subject) && is_string($subject)) {
      $subject = trim(strip_tags(stripslashes($subject)));
    } else {
      throw new LinkedInException('LinkedIn->message(): bad data passed, $subject must be of type string.');
    }
    if(!empty($body) && is_string($body)) {
      $body = trim(strip_tags(stripslashes($body)));
    } else {
      throw new LinkedInException('LinkedIn->message(): bad data passed, $body must be of type string.');
    }
    if(!is_array($recipients) || count($recipients) < 1) {
      // no recipients, and/or bad data
      throw new LinkedInException('LinkedIn->message(): at least one message recipient required.');
    }

    // construct the xml data
		$data   = '<?xml version="1.0" encoding="UTF-8"?>
		           <mailbox-item>
		             <recipients>';
    $data  .=     ($copy_self) ? '<recipient><person path="/people/~"/></recipient>' : '';
                  for($i = 0; $i < count($recipients); $i++) {
                    if(is_string($recipients[$i])) {
                      $data .= '<recipient><person path="/people/' . trim($recipients[$i]) . '"/></recipient>';
                    } else {
                      throw new LinkedInException ('LinkedIn->message(): bad data passed, $recipients must be an array of type string.');
                    }
                  }
    $data  .= '  </recipients>
                 <subject>' . htmlspecialchars($subject) . '</subject>
                 <body>' . htmlspecialchars($body) . '</body>
               </mailbox-item>';

    // send request
    $query    = self::_URL_API . '/v1/people/~/mailbox';
    $response = $this->fetch('POST', $query, $data);

		/**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * Job posting method.
	 *
	 * Post a job to LinkedIn, assuming that you have access to this feature.
	 * Full details from LinkedIn on this functionality can be found here:
	 *
	 *   http://developer.linkedin.com/community/jobs?view=documents
	 *
	 * @param str $xml
	 *    The XML defining a job to post.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function postJob($xml) {
    // check passed data
    if(is_string($xml)) {
      $xml = trim(stripslashes($xml));
    } else {
      throw new LinkedInException('LinkedIn->postJob(): bad data passed, $xml must be of type string.');
    }

    // construct and send the request
    $query    = self::_URL_API . '/v1/jobs';
    $response = $this->fetch('POST', $query, $xml);

		/**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * General profile retrieval function.
	 *
	 * Takes a string of parameters as input and requests profile data from the
	 * Linkedin Profile API. See the official documentation for $options
	 * 'field selector' formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1014
	 *   http://developer.linkedin.com/docs/DOC-1002
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function profile($options = '~') {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->profile(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

  /**
   * Send a message
   *
   * Send a message to another member or members
   *
   * @author @timersys
   *
   * @param array $recipients
   *    The id or ids to send the message to.
   * @param str $subject
   *    The subject of the invitation to send.
   * @param str $body
   *    The body of the invitation to send.
   *
   * @return arr array
   *    Containing retrieval success, LinkedIn response.
   */
   public function send_msg($recipients, $subject, $body) {
     /**
     	* Clean up the passed data per these rules:
     	*
     	* 1) No HTML permitted
     	* 2) 200 characters max in the invitation subject
     	* 3) Only able to connect as a friend at this point
     	*/

        // check passed data
        if(empty($recipients)) {
          throw new LinkedInException('LinkedIn->send_msg(): you must provide an invitation recipient.');
        }
        if(!empty($subject)) {
          $subject = trim(htmlspecialchars(strip_tags(stripslashes($subject))));
        } else {
          throw new LinkedInException('LinkedIn->send_msg(): message subject is empty.');
        }
        if(!empty($body)) {
          $body = trim(htmlspecialchars(strip_tags(stripslashes($body))));

          if(strlen($body) > self::_INV_BODY_LENGTH) {
           throw new LinkedInException('LinkedIn->send_msg(): message body length is too long - max length is ' . self::_INV_BODY_LENGTH . ' characters.');
          }
        } else {
          throw new LinkedInException('LinkedIn->send_msg(): message body is empty.');
        }

       // construct the xml data
       $data = '<?xml version="1.0" encoding="UTF-8"?>
       <mailbox-item>
       <recipients>';

       foreach( $recipients as $recipient )
       {
         $data .= '<recipient>';
         $data .= '<person path="/people/'. $recipient . '"/>';
         $data .= '</recipient>';
       }
       $data .= ' </recipients>
           <subject>' . $subject . '</subject>
           <body>' . $body . '</body>
           </mailbox-item>';
      // send request
      $query = self::_URL_API . '/v1/people/~/mailbox';
      $response = $this->fetch('POST', $query, $data);

     	/**
     	 * Check for successful request (a 201 response from LinkedIn server)
     	 * per the documentation linked in method comments above.
       */
      return $this->checkResponse(201, $response);
  }

	/**
	 * Manual API call method, allowing for support for un-implemented API
	 * functionality to be supported.
	 *
	 * @param str $method
	 *    The data communication method.
	 * @param str $url
	 *    The Linkedin API endpoint to connect with - should NOT include the
	 *    leading https://api.linkedin.com/v1.
	 * @param str $body
	 *    [OPTIONAL] The URL-encoded body data to send to LinkedIn with the request.
	 *
	 * @return arr
	 * 		array containing retrieval information, LinkedIn response. Note that you
	 * 		must manually check the return code and compare this to the expected
	 * 		API response to determine  if the raw call was successful.
	 */
	public function raw($method, $url, $body = null) {
	  if(!is_string($method)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->raw(): bad data passed, $method must be of string value.');
	  }
	  if(!is_string($url)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->raw(): bad data passed, $url must be of string value.');
	  }
	  if(!is_null($body) && !is_string($url)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->raw(): bad data passed, $body must be of string value.');
	  }

    // construct and send the request
	  $query = self::_URL_API . '/v1' . trim($url);
	  return $this->fetch($method, $query, $body);
	}

	/**
	 * This removes the specified group from the group suggestions, per:
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function removeSuggestedGroup($gid) {
		if(!is_string($gid)) {
			throw new LinkedInException('LinkedIn->removeSuggestedGroup(): bad data passed, $gid must be of type string');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/suggestions/groups/'  .trim($gid);
		$response = $this->fetch('DELETE', $query);

		/**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(204, $response);
	}

	/**
	 * Renew a job.
	 *
	 * Calling this method causes the passed job to be renewed, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1154
	 *
	 * @param str $jid
	 *    Job ID you want to renew.
	 * @param str $cid
	 *    Contract ID that covers the passed Job ID.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function renewJob($jid, $cid) {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->renewJob(): bad data passed, $jid must be of string value.');
	  }
	  if(!is_string($cid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->renewJob(): bad data passed, $cid must be of string value.');
	  }

	  // construct the xml data
		$data   = '<?xml version="1.0" encoding="UTF-8"?>
		           <job>
		             <contract-id>' . trim($cid) . '</contract-id>
                 <renewal/>
               </job>';

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/jobs/partner-job-id=' . trim($jid);
	  $response = $this->fetch('PUT', $query, $data);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

  /**
	 * Access token retrieval.
	 *
	 * Request the user's access token from the Linkedin API.
	 *
	 * @param str $token
	 *    The token returned from the user authorization stage.
	 * @param str $secret
	 *    The secret returned from the request token stage.
	 * @param str $verifier
	 *    The verification value from LinkedIn.
	 *
	 * @return arr
	 *    The Linkedin OAuth/http response, in array format.
	 */
	public function retrieveTokenAccess($token, $secret, $verifier) {
	  // check passed data
    if(!is_string($token) || !is_string($secret) || !is_string($verifier)) {
      // nothing passed, raise an exception
		  throw new LinkedInException('LinkedIn->retrieveTokenAccess(): bad data passed, string type is required for $token, $secret and $verifier.');
    }

    // start retrieval process
	  $this->setToken(array('oauth_token' => $token, 'oauth_token_secret' => $secret));
    $parameters = array(
      'oauth_verifier' => $verifier
    );
    $response = $this->fetch(self::_METHOD_TOKENS, self::_URL_ACCESS, null, $parameters);
    parse_str($response['linkedin'], $response['linkedin']);

    /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    if($response['info']['http_code'] == 200) {
      // tokens retrieved
      $this->setToken($response['linkedin']);

      // set the response
      $return_data            = $response;
      $return_data['success'] = true;
    } else {
      // error getting the request tokens
       $this->setToken(null);

      // set the response
      $return_data            = $response;
      $return_data['error']   = 'HTTP response from LinkedIn end-point was not code 200';
      $return_data['success'] = false;
    }
    return $return_data;
	}

	/**
	 * Request token retrieval.
	 *
	 * Get the request token from the Linkedin API.
	 *
	 * @return arr
	 *    The Linkedin OAuth/http response, in array format.
	 */
	public function retrieveTokenRequest() {
    $parameters = array(
      'oauth_callback' => $this->getCallbackUrl()
    );
    $response = $this->fetch(self::_METHOD_TOKENS, self::_URL_REQUEST, null, $parameters);
    parse_str($response['linkedin'], $response['linkedin']);

    /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    if(($response['info']['http_code'] == 200) && (array_key_exists('oauth_callback_confirmed', $response['linkedin'])) && ($response['linkedin']['oauth_callback_confirmed'] == 'true')) {
      // tokens retrieved
      $this->setToken($response['linkedin']);

      // set the response
      $return_data            = $response;
      $return_data['success'] = true;
    } else {
      // error getting the request tokens
      $this->setToken(null);

      // set the response
      $return_data = $response;
      if((array_key_exists('oauth_callback_confirmed', $response['linkedin'])) && ($response['linkedin']['oauth_callback_confirmed'] == 'true')) {
        $return_data['error'] = 'HTTP response from LinkedIn end-point was not code 200';
      } else {
        $return_data['error'] = 'OAuth callback URL was not confirmed by the LinkedIn end-point';
      }
      $return_data['success'] = false;
    }
    return $return_data;
	}

	/**
	 * User authorization revocation.
	 *
	 * Revoke the current user's access token, clear the access token's from
	 * current LinkedIn object. The current documentation for this feature is
	 * found in a blog entry from April 29th, 2010:
	 *
	 *   http://developer.linkedin.com/community/apis/blog/2010/04/29/oauth--now-for-authentication
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function revoke() {
	  // construct and send the request
	  $response = $this->fetch('GET', self::_URL_REVOKE);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(200, $response);
	}

	/**
	 * [DEPRECATED] General people search function.
	 *
	 * Takes a string of parameters as input and requests profile data from the
	 * Linkedin People Search API.  See the official documentation for $options
	 * querystring formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1191
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function search($options = null) {
		return searchPeople($options);
	}

	/**
	 * Company search.
	 *
	 * Uses the Company Search API to find companies using keywords, industry,
	 * location, or some other criteria. It returns a collection of matching
	 * companies.
	 *
	 *   http://developer.linkedin.com/docs/DOC-1325
	 *
	 * @param str $options
	 *    [OPTIONAL] Search options.
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function searchCompanies($options = '') {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->searchCompanies(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/company-search' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Jobs search.
	 *
	 * Use the Job Search API to find jobs using keywords, company, location,
	 * or some other criteria. It returns a collection of matching jobs. Each
	 * entry can contain much of the information available on the job listing.
	 *
	 *   http://developer.linkedin.com/docs/DOC-1321
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function searchJobs($options = '') {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->jobsSearch(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/job-search' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * General people search function.
	 *
	 * Takes a string of parameters as input and requests profile data from the
	 * Linkedin People Search API.  See the official documentation for $options
	 * querystring formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1191
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function searchPeople($options = null) {
	  // check passed data
    if(!is_null($options) && !is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->search(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
    $query    = self::_URL_API . '/v1/people-search' . trim($options);
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * Set the application_key property.
	 *
	 * @param str $key
	 *    The application key.
	 */
	public function setApplicationKey($key) {
	  $this->application_key = $key;
	}

	/**
	 * Set the application_secret property.
	 *
	 * @param str $secret
	 *    The application secret.
	 */
	public function setApplicationSecret($secret) {
	  $this->application_secret = $secret;
	}

	/**
	 * Set the callback property.
	 *
	 * @param str $url
	 *    The callback url.
	 */
	public function setCallbackUrl($url) {
	  $this->callback = $url;
	}

	/**
	 * This sets the group settings of the specified group.
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @param str $gid
	 * 		The group id.
	 * @param str $xml
	 * 		The group settings to set. The settings are:
	 * 		  -<show-group-logo-in-profile>
	 * 		  -<contact-email>
	 * 		  -<email-digest-frequency>
	 * 		  -<email-announcements-from-managers>
	 * 		  -<allow-messages-from-members>
	 * 		  -<email-for-every-new-post>
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function setGroupSettings($gid, $xml) {
		if(!is_string ($gid)) {
      throw new LinkedInException('LinkedIn->setGroupSettings(): bad data passed, $token_access should be in array format.');
		}
		if(!is_string ($xml)) {
      throw new LinkedInException('LinkedIn->setGroupSettings(): bad data passed, $token_access should be in array format.');
		}

		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/group-memberships/' . trim($gid);
		$response = $this->fetch('PUT', $query, $xml);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * Set the response_format property.
	 *
	 * @param str $format
	 *    [OPTIONAL] The response format to specify to LinkedIn.
	 */
	public function setResponseFormat($format = self::_DEFAULT_RESPONSE_FORMAT) {
	  $this->response_format = $format;
	}

	/**
	 * Set the token property.
	 *
	 * @return arr $token
	 *    The LinkedIn OAuth token.
	 */
	public function setToken($token) {
    // check passed data
    if(!is_null($token) && !is_array($token)) {
      // bad data passed
      throw new LinkedInException('LinkedIn->setToken(): bad data passed, $token_access should be in array format.');
    }

    // set token
    $this->token = $token;
	}

	/**
	 * [DEPRECATED] Set the token_access property.
	 *
	 * @return arr $token_access
	 *    [OPTIONAL] The LinkedIn OAuth access token.
	 */
	public function setTokenAccess($token_access) {
    $this->setToken($token_access);
	}

	/**
	 * Post a share.
	 *
	 * Create a new or reshare another user's shared content. Full details from
	 * LinkedIn on this functionality can be found here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1212
	 *
	 *   $action values: ('new', 'reshare')
	 *   $content format:
	 *     $action = 'new'; $content => ('comment' => 'xxx', 'title' => 'xxx', 'submitted-url' => 'xxx', 'submitted-image-url' => 'xxx', 'description' => 'xxx')
	 *     $action = 'reshare'; $content => ('comment' => 'xxx', 'id' => 'xxx')
	 *
	 * @param str $action
	 *    The sharing action to perform.
	 * @param str $content
	 *    The share content.
	 * @param bool $private
	 *    [OPTIONAL] Should we restrict this shared item to connections only?
	 * @param bool $twitter
	 *    [OPTIONAL] Also update the teathered Twitter account.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function share($action, $content, $private = true, $twitter = false) {
	  // check the status itself
    if(!empty($action) && !empty($content)) {
      /**
       * Status is not empty, wrap a cleaned version of it in xml.  Status
       * rules:
       *
       * 1) Comments are 700 chars max (if this changes, change _SHARE_COMMENT_LENGTH constant)
       * 2) Content/title 200 chars max (if this changes, change _SHARE_CONTENT_TITLE_LENGTH constant)
       * 3) Content/description 400 chars max (if this changes, change _SHARE_CONTENT_DESC_LENGTH constant)
       * 4a) New shares must contain a comment and/or (content/title and content/submitted-url)
       * 4b) Reshared content must contain an attribution id.
       * 4c) Reshared content must contain actual content, not just a comment.
       * 5) No HTML permitted in comment, content/title, content/description.
       */

      // prepare the share data per the rules above
      $share_flag   = false;
      $content_xml  = null;
      switch($action) {
        case 'new':
          // share can be an article
          if(array_key_exists('title', $content) && array_key_exists('submitted-url', $content)) {
            // we have shared content, format it as needed per rules above
            $content_title = trim(htmlspecialchars(strip_tags(stripslashes($content['title']))));
            if(strlen($content_title) > self::_SHARE_CONTENT_TITLE_LENGTH) {
              throw new LinkedInException('LinkedIn->share(): title length is too long - max length is ' . self::_SHARE_CONTENT_TITLE_LENGTH . ' characters.');
            }
            $content_xml .= '<content>
                               <title>' . $content_title . '</title>
                               <submitted-url>' . trim(htmlspecialchars($content['submitted-url'])) . '</submitted-url>';
            if(array_key_exists('submitted-image-url', $content)) {
              $content_xml .= '<submitted-image-url>' . trim(htmlspecialchars($content['submitted-image-url'])) . '</submitted-image-url>';
            }
            if(array_key_exists('description', $content)) {
              $content_desc = trim(htmlspecialchars(strip_tags(stripslashes($content['description']))));
              if(strlen($content_desc) > self::_SHARE_CONTENT_DESC_LENGTH) {
                throw new LinkedInException('LinkedIn->share(): description length is too long - max length is ' . self::_SHARE_CONTENT_DESC_LENGTH . ' characters.');
              }
              $content_xml .= '<description>' . $content_desc . '</description>';
            }
            $content_xml .= '</content>';

            $share_flag = true;
          }

          // share can be just a comment
          if(array_key_exists('comment', $content)) {
          	// comment located
          	$comment = htmlspecialchars(trim(strip_tags(stripslashes($content['comment']))));
          	if(strlen($comment) > self::_SHARE_COMMENT_LENGTH) {
              throw new LinkedInException('LinkedIn->share(): comment length is too long - max length is ' . self::_SHARE_COMMENT_LENGTH . ' characters.');
            }
            $content_xml .= '<comment>' . $comment . '</comment>';

          	$share_flag = true;
      	  }
          break;
        case 'reshare':
          if(array_key_exists('id', $content)) {
            // put together the re-share attribution XML
            $content_xml .= '<attribution>
                               <share>
                                 <id>' . trim($content['id']) . '</id>
                               </share>
                             </attribution>';

            // optional additional comment
            if(array_key_exists('comment', $content)) {
            	// comment located
            	$comment = htmlspecialchars(trim(strip_tags(stripslashes($content['comment']))));
            	if(strlen($comment) > self::_SHARE_COMMENT_LENGTH) {
                throw new LinkedInException('LinkedIn->share(): comment length is too long - max length is ' . self::_SHARE_COMMENT_LENGTH . ' characters.');
              }
              $content_xml .= '<comment>' . $comment . '</comment>';
        	  }

        	  $share_flag = true;
          }
          break;
        default:
          // bad action passed
          throw new LinkedInException('LinkedIn->share(): share action is an invalid value, must be one of: share, reshare.');
          break;
      }

      // should we proceed?
      if($share_flag) {
        // put all of the xml together
        $visibility = ($private) ? 'connections-only' : 'anyone';
        $data       = '<?xml version="1.0" encoding="UTF-8"?>
                       <share>
                         ' . $content_xml . '
                         <visibility>
                           <code>' . $visibility . '</code>
                         </visibility>
                       </share>';

        // create the proper url
        $share_url = self::_URL_API . '/v1/people/~/shares';
  		  if($twitter) {
  			  // update twitter as well
          $share_url .= '?twitter-post=true';
  			}

        // send request
        $response = $this->fetch('POST', $share_url, $data);
  		} else {
  		  // data constraints/rules not met, raise an exception
		    throw new LinkedInException('LinkedIn->share(): sharing data constraints not met; check that you have supplied valid content and combinations of content to share.');
  		}
    } else {
      // data missing, raise an exception
		  throw new LinkedInException('LinkedIn->share(): sharing action or shared content is missing.');
    }

    /**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * Network statistics.
	 *
	 * General network statistics retrieval function, returns the number of connections,
	 * second-connections an authenticated user has. More information here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1006
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function statistics() {
	  // construct and send the request
    $query    = self::_URL_API . '/v1/people/~/network/network-stats';
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse(200, $response);
	}

	/**
	 * Companies you may want to follow.
	 *
	 * Returns a list of companies the current user may want to follow, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1324
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function suggestedCompanies() {
	  // construct and send the request
    $query    = self::_URL_API . '/v1/people/~/suggestions/to-follow/companies';
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Retrieves suggested groups for the user, per:
	 *
	 *   http://developer.linkedin.com/documents/groups-api
	 *
	 * @return arr
	 * 		array containing retrieval success, LinkedIn response.
	 */
	public function suggestedGroups() {
		// construct and send the request
		$query    = self::_URL_API . '/v1/people/~/suggestions/groups:(id,name,is-open-to-non-members)';
		$response = $this->fetch('GET', $query);

		/**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
		return $this->checkResponse (200, $response);
	}

	/**
	 * Jobs you may be interested in.
	 *
	 * Returns a list of jobs the current user may be interested in, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1323
	 *
 	 * @param str $options
 	 *    [OPTIONAL] Data retrieval options.
 	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function suggestedJobs($options = ':(jobs)') {
	  // check passed data
	  if(!is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->suggestedJobs(): bad data passed, $options must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/suggestions/job-suggestions' . trim($options);
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Unbookmark a job.
	 *
	 * Calling this method causes the current user to remove a bookmark for the
	 * specified job:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1323
	 *
	 * @param str $jid
	 *    Job ID you want to unbookmark.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function unbookmarkJob($jid) {
	  // check passed data
	  if(!is_string($jid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->unbookmarkJob(): bad data passed, $jid must be of type string.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/job-bookmarks/' . trim($jid);
	  $response = $this->fetch('DELETE', $query);

	  /**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(204, $response);
	}

	/**
	 * Unfollow a company.
	 *
	 * Calling this method causes the current user to stop following the specified
	 * company, per:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1324
	 *
	 * @param str $cid
	 *    Company ID you want to unfollow.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function unfollowCompany($cid) {
	  // check passed data
	  if(!is_string($cid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->unfollowCompany(): bad data passed, $cid must be of string value.');
	  }

	  // construct and send the request
	  $query    = self::_URL_API . '/v1/people/~/following/companies/id=' . trim($cid);
	  $response = $this->fetch('DELETE', $query);

	  /**
	   * Check for successful request (a 204 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(204, $response);
	}

	/**
	 * Unlike a network update.
	 *
	 * Unlike another user's network update:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $uid
	 *    The LinkedIn update ID.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function unlike($uid) {
	  // check passed data
	  if(!is_string($uid)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->unlike(): bad data passed, $uid must be of type string.');
	  }

    // construct the xml data
		$data = '<?xml version="1.0" encoding="UTF-8"?>
		         <is-liked>false</is-liked>';

		// send request
    $query    = self::_URL_API . '/v1/people/~/network/updates/key=' . $uid . '/is-liked';
    $response = $this->fetch('PUT', $query, $data);

  	/**
	   * Check for successful request (a 201 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
    return $this->checkResponse(201, $response);
	}

	/**
	 * Post network update.
	 *
	 * Update the user's Linkedin network status. Full details from LinkedIn
	 * on this functionality can be found here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1009
	 *   http://developer.linkedin.com/docs/DOC-1009#comment-1077
	 *
	 * @param str $update
	 *    The network update.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function updateNetwork($update) {
	  // check passed data
    if(!is_string($update)) {
      // nothing/non-string passed, raise an exception
		  throw new LinkedInException('LinkedIn->updateNetwork(): bad data passed, $update must be a non-zero length string.');
    }

    /**
     * Network update is not empty, wrap a cleaned version of it in xml.
     * Network update rules:
     *
     * 1) No HTML permitted except those found in _NETWORK_HTML constant
     * 2) Update cannot be longer than 140 characters.
     */
    // get the user data
    $response = self::profile('~:(first-name,last-name,site-standard-profile-request)');
    if($response['success'] === true) {
      /**
       * We are converting response to usable data.  I'd use SimpleXML here, but
       * to keep the class self-contained, we will use a portable XML parsing
       * routine, self::xmlToarray.
       */
      $person = self::xmlToarray($response['linkedin']);
      if($person === false) {
        // bad xml data
        throw new LinkedInException('LinkedIn->updateNetwork(): LinkedIn returned bad XML data.');
      }
  		$fields = $person['person']['children'];

  		// prepare user data
  		$first_name   = trim($fields['first-name']['content']);
  		$last_name    = trim($fields['last-name']['content']);
  		$profile_url  = trim($fields['site-standard-profile-request']['children']['url']['content']);

      // create the network update
      $update = trim(htmlspecialchars(strip_tags($update, self::_NETWORK_HTML)));
      if(strlen($update) > self::_NETWORK_LENGTH) {
        throw new LinkedInException('LinkedIn->share(): update length is too long - max length is ' . self::_NETWORK_LENGTH . ' characters.');
      }
      $user   = htmlspecialchars('<a href="' . $profile_url . '">' . $first_name . ' ' . $last_name . '</a>');
  		$data   = '<activity locale="en_US">
    				       <content-type>linkedin-html</content-type>
    				       <body>' . $user . ' ' . $update . '</body>
    				     </activity>';

      // send request
      $query    = self::_URL_API . '/v1/people/~/person-activities';
      $response = $this->fetch('POST', $query, $data);

      /**
  	   * Check for successful request (a 201 response from LinkedIn server)
  	   * per the documentation linked in method comments above.
  	   */
      return $this->checkResponse(201, $response);
    } else {
      // profile retrieval failed
      throw new LinkedInException('LinkedIn->updateNetwork(): profile data could not be retrieved.');
    }
	}

  /**
	 * General network update retrieval function.
	 *
	 * Takes a string of parameters as input and requests update-related data
	 * from the Linkedin Network Updates API. See the official documentation for
	 * $options parameter formatting:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1006
	 *
	 * For getting more comments, likes, etc, see here:
	 *
	 *   http://developer.linkedin.com/docs/DOC-1043
	 *
	 * @param str $options
	 *    [OPTIONAL] Data retrieval options.
	 * @param str $id
	 *    [OPTIONAL] The LinkedIn ID to restrict the updates for.
	 *
	 * @return arr
	 *    array containing retrieval success, LinkedIn response.
	 */
	public function updates($options = null, $id = null) {
	  // check passed data
    if(!is_null($options) && !is_string($options)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->updates(): bad data passed, $options must be of type string.');
	  }
	  if(!is_null($id) && !is_string($id)) {
	    // bad data passed
		  throw new LinkedInException('LinkedIn->updates(): bad data passed, $id must be of type string.');
	  }

	  // construct and send the request
	  if(!is_null($id) && self::isId($id)) {
	    $query = self::_URL_API . '/v1/people/' . $id . '/network/updates' . trim($options);
	  } else {
      $query = self::_URL_API . '/v1/people/~/network/updates' . trim($options);
    }
	  $response = $this->fetch('GET', $query);

	  /**
	   * Check for successful request (a 200 response from LinkedIn server)
	   * per the documentation linked in method comments above.
	   */
	  return $this->checkResponse(200, $response);
	}

	/**
	 * Converts passed XML data to an array.
	 *
	 * @param str $xml
	 *    The XML to convert to an array.
	 *
	 * @return arr
	 *    array containing the XML data.
	 * @return bool
	 *    false if passed data cannot be parsed to an array.
	 */
	public static function xmlToarray($xml) {
	  // check passed data
    if(!is_string($xml)) {
	    // bad data passed
      throw new LinkedInException('LinkedIn->xmlToarray(): bad data passed, $xml must be a non-zero length string.');
	  }

	  $parser = xml_parser_create();
	  xml_parser_set_option($parser, XML_OPTION_CASE_FOLDING, 0);
    xml_parser_set_option($parser, XML_OPTION_SKIP_WHITE, 1);
    if(xml_parse_into_struct($parser, $xml, $tags)) {
	    $elements = array();
      $stack    = array();
      foreach($tags as $tag) {
        $index = count($elements);
        if($tag['type'] == 'complete' || $tag['type'] == 'open') {
          $elements[$tag['tag']]               = array();
          $elements[$tag['tag']]['attributes'] = (array_key_exists('attributes', $tag)) ? $tag['attributes'] : null;
          $elements[$tag['tag']]['content']    = (array_key_exists('value', $tag)) ? $tag['value'] : null;
          if($tag['type'] == 'open') {
            $elements[$tag['tag']]['children'] = array();
            $stack[count($stack)] = &$elements;
            $elements = &$elements[$tag['tag']]['children'];
          }
        }
        if($tag['type'] == 'close') {
          $elements = &$stack[count($stack) - 1];
          unset($stack[count($stack) - 1]);
        }
      }
      $return_data = $elements;
	  } else {
	    // not valid xml data
	    $return_data = false;
	  }
	  xml_parser_free($parser);
    return $return_data;
  }
}