[AutoJMBridge] Fix bridge after website change (#1081)
* [AutoJMBridge] Fix bridge after website change The website was totally reworked, so the bridge had to be reworked too. The bridge parameters changed, therefore old RSS feed will not work anymore, but it was impossible to do it in another way.
This commit is contained in:
parent
b6943de0ca
commit
291e8c2a23
1 changed files with 157 additions and 37 deletions
|
@ -3,63 +3,183 @@
|
||||||
class AutoJMBridge extends BridgeAbstract {
|
class AutoJMBridge extends BridgeAbstract {
|
||||||
|
|
||||||
const NAME = 'AutoJM';
|
const NAME = 'AutoJM';
|
||||||
const URI = 'http://www.autojm.fr/';
|
const URI = 'https://www.autojm.fr/';
|
||||||
const DESCRIPTION = 'Suivre les offres de véhicules proposés par AutoJM en fonction des critères de filtrages';
|
const DESCRIPTION = 'Suivre les offres de véhicules proposés par AutoJM en fonction des critères de filtrages';
|
||||||
const MAINTAINER = 'sysadminstory';
|
const MAINTAINER = 'sysadminstory';
|
||||||
const PARAMETERS = array(
|
const PARAMETERS = array(
|
||||||
'Afficher les offres de véhicules disponible en fonction des critères du site AutoJM' => array(
|
'Afficher les offres de véhicules disponible en fonction des critères du site AutoJM' => array(
|
||||||
'url' => array(
|
'url' => array(
|
||||||
'name' => 'URL de la recherche',
|
'name' => 'URL du modèle',
|
||||||
'type' => 'text',
|
'type' => 'text',
|
||||||
'required' => true,
|
'required' => true,
|
||||||
'title' => 'URL d\'une recherche avec filtre de véhicules sans le http://www.autojm.fr/',
|
'title' => 'URL d\'une recherche avec filtre de véhicules sans le http://www.autojm.fr/',
|
||||||
'exampleValue' => 'gammes/index/398?order_by=finition_asc&energie[]=3&transmission[]=2&dispo=all'
|
'exampleValue' => 'achat-voitures-neuves-peugeot-nouvelle-308-5p'
|
||||||
|
),
|
||||||
|
'isDispo' => array(
|
||||||
|
'name' => 'Disponibilité',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'-' => '',
|
||||||
|
'En stock' => 1,
|
||||||
|
'Sur commande' => 0
|
||||||
|
),
|
||||||
|
'title' => 'Critère de disponibilité'
|
||||||
|
),
|
||||||
|
'energy' => array(
|
||||||
|
'name' => 'Carburant',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'-' => '',
|
||||||
|
'Diesel' => 1,
|
||||||
|
'Essence' => 3,
|
||||||
|
'Hybride' => 5
|
||||||
|
),
|
||||||
|
'title' => 'Carburant'
|
||||||
|
),
|
||||||
|
'transmission' => array(
|
||||||
|
'name' => 'Transmission',
|
||||||
|
'type' => 'list',
|
||||||
|
'values' => array(
|
||||||
|
'-' => '',
|
||||||
|
'Automatique' => 1,
|
||||||
|
'Manuelle' => 2
|
||||||
|
),
|
||||||
|
'title' => 'Transmission'
|
||||||
|
),
|
||||||
|
'priceMin' => array(
|
||||||
|
'name' => 'Prix minimum',
|
||||||
|
'type' => 'number',
|
||||||
|
'required' => false,
|
||||||
|
'title' => 'Prix minimum du véhicule',
|
||||||
|
'exampleValue' => '10000',
|
||||||
|
'defaultValue' => '0'
|
||||||
|
),
|
||||||
|
'priceMax' => array(
|
||||||
|
'name' => 'Prix maximum',
|
||||||
|
'type' => 'number',
|
||||||
|
'required' => false,
|
||||||
|
'title' => 'Prix maximum du véhicule',
|
||||||
|
'exampleValue' => '15000',
|
||||||
|
'defaultValue' => '150000'
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
const CACHE_TIMEOUT = 3600;
|
const CACHE_TIMEOUT = 3600;
|
||||||
|
|
||||||
public function getIcon() {
|
public function getIcon() {
|
||||||
return self::URI . 'assets/images/favicon.ico';
|
return self::URI . 'favicon.ico';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function collectData() {
|
public function collectData() {
|
||||||
$html = getSimpleHTMLDOM(self::URI . $this->getInput('url'))
|
|
||||||
or returnServerError('Could not request AutoJM.');
|
|
||||||
$list = $html->find('div[class*=ligne_modele]');
|
|
||||||
foreach($list as $element) {
|
|
||||||
$image = $element->find('img[class=width-100]', 0)->src;
|
|
||||||
$serie = $element->find('div[class=serie]', 0)->find('span', 0)->plaintext;
|
|
||||||
$url = $element->find('div[class=serie]', 0)->find('a[class=btn_ligne color-black]', 0)->href;
|
|
||||||
if($element->find('div[class*=hasStock-info]', 0) != null) {
|
|
||||||
$dispo = 'Disponible';
|
|
||||||
} else {
|
|
||||||
$dispo = 'Sur commande';
|
|
||||||
}
|
|
||||||
$carburant = str_replace('dispo |', '', $element->find('div[class=carburant]', 0)->plaintext);
|
|
||||||
$transmission = $element->find('div[class*=bv]', 0)->plaintext;
|
|
||||||
$places = $element->find('div[class*=places]', 0)->plaintext;
|
|
||||||
$portes = $element->find('div[class*=nb_portes]', 0)->plaintext;
|
|
||||||
$carosserie = $element->find('div[class*=coloris]', 0)->plaintext;
|
|
||||||
$remise = $element->find('div[class*=remise]', 0)->plaintext;
|
|
||||||
$prix = $element->find('div[class*=prixjm]', 0)->plaintext;
|
|
||||||
|
|
||||||
|
$model_url = self::URI . $this->getInput('url');
|
||||||
|
|
||||||
|
// Get the session cookies and the form token
|
||||||
|
$this->getInitialParameters($model_url);
|
||||||
|
|
||||||
|
// Build the form
|
||||||
|
$post_data = array(
|
||||||
|
'form[isDispo]' => $this->getInput('isDispo'),
|
||||||
|
'form[energy]' => $this->getInput('energy'),
|
||||||
|
'form[transmission]' => $this->getInput('transmission'),
|
||||||
|
'form[priceMin]' => $this->getInput('priceMin'),
|
||||||
|
'form[priceMin]' => $this->getInput('priceMin'),
|
||||||
|
'form[_token]' => $this->token
|
||||||
|
);
|
||||||
|
|
||||||
|
// Set the Form request content type
|
||||||
|
$header = array(
|
||||||
|
'Content-Type: application/x-www-form-urlencoded; charset=UTF-8',
|
||||||
|
);
|
||||||
|
|
||||||
|
// Set the curl options (POST query and content, and session cookies
|
||||||
|
$curl_opts = array(
|
||||||
|
CURLOPT_POST => true,
|
||||||
|
CURLOPT_POSTFIELDS => http_build_query($post_data),
|
||||||
|
CURLOPT_COOKIE => $this->cookies
|
||||||
|
);
|
||||||
|
|
||||||
|
// Get the JSON content of the form
|
||||||
|
$json = getContents($model_url, $header, $curl_opts)
|
||||||
|
or returnServerError('Could not request AutoJM.');
|
||||||
|
|
||||||
|
// Extract the HTML content from the JSON result
|
||||||
|
$data = json_decode($json);
|
||||||
|
$html = str_get_html($data->content);
|
||||||
|
|
||||||
|
// Go through every finisha of the model
|
||||||
|
$list = $html->find('h2');
|
||||||
|
foreach ($list as $finish) {
|
||||||
|
$finish_name = $finish->plaintext;
|
||||||
|
$motorizations = $finish->next_sibling()->find('li');
|
||||||
|
foreach ($motorizations as $element) {
|
||||||
|
$image = $element->find('div[class=block-product-image]', 0)->{'data-ga-banner'};
|
||||||
|
$serie = $element->find('span[class=model]', 0)->plaintext;
|
||||||
|
$url = self::URI . substr($element->find('a', 0)->href, 1);
|
||||||
|
if ($element->find('span[class*=block-product-nbModel]', 0) != null) {
|
||||||
|
$availability = 'En Stock';
|
||||||
|
} else {
|
||||||
|
$availability = 'Sur commande';
|
||||||
|
}
|
||||||
|
$discount_html = $element->find('span[class*=tag--promo]', 0);
|
||||||
|
if ($discount_html != null) {
|
||||||
|
$discount = $discount_html->plaintext;
|
||||||
|
} else {
|
||||||
|
$discount = 'inconnue';
|
||||||
|
}
|
||||||
|
$price = $element->find('span[class=price red h1]', 0)->plaintext;
|
||||||
$item = array();
|
$item = array();
|
||||||
$item['uri'] = $url;
|
$item['title'] = $finish_name . ' ' . $serie;
|
||||||
$item['title'] = $serie;
|
$item['content'] = '<p><img style="vertical-align:middle ; padding: 10px" src="' . $image . '" />'
|
||||||
$item['content'] = '<p><img style="vertical-align:middle ; padding: 10px" src="' . $image . '" />' . $serie . '</p>';
|
. $finish_name . ' ' . $serie . '</p>';
|
||||||
$item['content'] .= '<ul><li>Disponibilité : ' . $dispo . '</li>';
|
$item['content'] .= '<ul><li>Disponibilité : ' . $availability . '</li>';
|
||||||
$item['content'] .= '<li>Carburant : ' . $carburant . '</li>';
|
|
||||||
$item['content'] .= '<li>Transmission : ' . $transmission . '</li>';
|
|
||||||
$item['content'] .= '<li>Nombre de places : ' . $places . '</li>';
|
|
||||||
$item['content'] .= '<li>Nombre de portes : ' . $portes . '</li>';
|
|
||||||
$item['content'] .= '<li>Série : ' . $serie . '</li>';
|
$item['content'] .= '<li>Série : ' . $serie . '</li>';
|
||||||
$item['content'] .= '<li>Carosserie : ' . $carosserie . '</li>';
|
$item['content'] .= '<li>Remise : ' . $discount . '</li>';
|
||||||
$item['content'] .= '<li>Remise : ' . $remise . '</li>';
|
$item['content'] .= '<li>Prix : ' . $price . '</li></ul>';
|
||||||
$item['content'] .= '<li>Prix : ' . $prix . '</li></ul>';
|
|
||||||
|
// Add a fictionnal anchor to the RSS element URL, based on the item content ;
|
||||||
|
// As the URL could be identical even if the price change, some RSS reader will not show those offers as new items
|
||||||
|
$item['uri'] = $url . '#' . md5($item['content']);
|
||||||
|
|
||||||
$this->items[] = $item;
|
$this->items[] = $item;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the session cookie and the form token
|
||||||
|
*
|
||||||
|
* @param string $pageURL The URL from which to get the values
|
||||||
|
*/
|
||||||
|
private function getInitialParameters($pageURL) {
|
||||||
|
$ch = curl_init();
|
||||||
|
curl_setopt($ch, CURLOPT_URL, $pageURL);
|
||||||
|
curl_setopt($ch, CURLOPT_HEADER, true);
|
||||||
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||||
|
$data = curl_exec($ch);
|
||||||
|
|
||||||
|
// Separate the response header and the content
|
||||||
|
$headerSize = curl_getinfo($ch, CURLINFO_HEADER_SIZE);
|
||||||
|
$header = substr($data, 0, $headerSize);
|
||||||
|
$content = substr($data, $headerSize);
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
// Extract the cookies from the headers
|
||||||
|
$cookies = '';
|
||||||
|
$http_response_header = explode("\r\n", $header);
|
||||||
|
foreach ($http_response_header as $hdr) {
|
||||||
|
if (strpos($hdr, 'Set-Cookie') !== false) {
|
||||||
|
$cLine = explode(':', $hdr)[1];
|
||||||
|
$cLine = explode(';', $cLine)[0];
|
||||||
|
$cookies .= ';' . $cLine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$this->cookies = trim(substr($cookies, 1));
|
||||||
|
|
||||||
|
// Get the token from the content
|
||||||
|
$html = str_get_html($content);
|
||||||
|
$token = $html->find('input[type=hidden][id=form__token]', 0);
|
||||||
|
$this->token = $token->value;
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue