mirror of
https://github.com/Part-DB/Part-DB-server.git
synced 2025-06-21 01:25:55 +02:00
Added very basic octopart info provider
This commit is contained in:
parent
6a00b8e168
commit
43cc37d10f
6 changed files with 260 additions and 2 deletions
|
@ -21,3 +21,18 @@ knpu_oauth2_client:
|
||||||
#urlAuthorize: 'https://sandbox-api.digikey.com/v1/oauth2/authorize'
|
#urlAuthorize: 'https://sandbox-api.digikey.com/v1/oauth2/authorize'
|
||||||
#urlAccessToken: 'https://sandbox-api.digikey.com/v1/oauth2/token'
|
#urlAccessToken: 'https://sandbox-api.digikey.com/v1/oauth2/token'
|
||||||
#urlResourceOwnerDetails: ''
|
#urlResourceOwnerDetails: ''
|
||||||
|
|
||||||
|
ip_octopart_oauth:
|
||||||
|
type: generic
|
||||||
|
provider_class: '\League\OAuth2\Client\Provider\GenericProvider'
|
||||||
|
|
||||||
|
client_id: '%env(PROVIDER_OCTOPART_CLIENT_ID)%'
|
||||||
|
client_secret: '%env(PROVIDER_OCTOPART_SECRET)%'
|
||||||
|
|
||||||
|
redirect_route: 'oauth_client_check'
|
||||||
|
redirect_params: { name: 'ip_octopart_oauth' }
|
||||||
|
|
||||||
|
provider_options:
|
||||||
|
urlAuthorize: 'https://identity.nexar.com/connect/authorize'
|
||||||
|
urlAccessToken: 'https://identity.nexar.com/connect/token'
|
||||||
|
urlResourceOwnerDetails: ''
|
|
@ -16,8 +16,9 @@ nelmio_security:
|
||||||
# Whitelist the domain of the SAML IDP, so we can redirect to it during the SAML login process
|
# Whitelist the domain of the SAML IDP, so we can redirect to it during the SAML login process
|
||||||
- '%env(string:key:host:url:SAML_IDP_SINGLE_SIGN_ON_SERVICE)%'
|
- '%env(string:key:host:url:SAML_IDP_SINGLE_SIGN_ON_SERVICE)%'
|
||||||
|
|
||||||
# Whitelist the info provider APIs
|
# Whitelist the info provider APIs (OAuth redirects)
|
||||||
- 'digikey.com'
|
- 'digikey.com'
|
||||||
|
- 'nexar.com'
|
||||||
|
|
||||||
# forces Microsoft's XSS-Protection with
|
# forces Microsoft's XSS-Protection with
|
||||||
# its block mode
|
# its block mode
|
||||||
|
|
|
@ -53,6 +53,8 @@ class PartDetailDTO extends SearchResultDTO
|
||||||
public readonly ?array $vendor_infos = null,
|
public readonly ?array $vendor_infos = null,
|
||||||
/** The mass of the product in grams */
|
/** The mass of the product in grams */
|
||||||
public readonly ?float $mass = null,
|
public readonly ?float $mass = null,
|
||||||
|
/** The URL to the product on the website of the manufacturer */
|
||||||
|
public readonly ?string $manufacturer_product_url = null,
|
||||||
) {
|
) {
|
||||||
parent::__construct(
|
parent::__construct(
|
||||||
provider_key: $provider_key,
|
provider_key: $provider_key,
|
||||||
|
|
|
@ -162,6 +162,7 @@ final class DTOtoEntityConverter
|
||||||
|
|
||||||
$entity->setManufacturerProductNumber($dto->mpn ?? '');
|
$entity->setManufacturerProductNumber($dto->mpn ?? '');
|
||||||
$entity->setManufacturingStatus($dto->manufacturing_status ?? ManufacturingStatus::NOT_SET);
|
$entity->setManufacturingStatus($dto->manufacturing_status ?? ManufacturingStatus::NOT_SET);
|
||||||
|
$entity->setManufacturerProductURL($dto->manufacturer_product_url ?? '');
|
||||||
|
|
||||||
//Set the provider reference on the part
|
//Set the provider reference on the part
|
||||||
$entity->setProviderReference(InfoProviderReference::fromPartDTO($dto));
|
$entity->setProviderReference(InfoProviderReference::fromPartDTO($dto));
|
||||||
|
|
|
@ -74,7 +74,7 @@ final class PartInfoRetriever
|
||||||
protected function searchInProvider(InfoProviderInterface $provider, string $keyword): array
|
protected function searchInProvider(InfoProviderInterface $provider, string $keyword): array
|
||||||
{
|
{
|
||||||
//Generate key and escape reserved characters from the provider id
|
//Generate key and escape reserved characters from the provider id
|
||||||
$escaped_keyword = urlencode($keyword);
|
$escaped_keyword = urlencode($keyword) . uniqid();
|
||||||
|
|
||||||
return $this->partInfoCache->get("search_{$provider->getProviderKey()}_{$escaped_keyword}", function (ItemInterface $item) use ($provider, $keyword) {
|
return $this->partInfoCache->get("search_{$provider->getProviderKey()}_{$escaped_keyword}", function (ItemInterface $item) use ($provider, $keyword) {
|
||||||
//Set the expiration time
|
//Set the expiration time
|
||||||
|
|
239
src/Services/InfoProviderSystem/Providers/OctopartProvider.php
Normal file
239
src/Services/InfoProviderSystem/Providers/OctopartProvider.php
Normal file
|
@ -0,0 +1,239 @@
|
||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* This file is part of Part-DB (https://github.com/Part-DB/Part-DB-symfony).
|
||||||
|
*
|
||||||
|
* Copyright (C) 2019 - 2023 Jan Böhmer (https://github.com/jbtronics)
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
|
||||||
|
namespace App\Services\InfoProviderSystem\Providers;
|
||||||
|
|
||||||
|
use App\Entity\Parts\ManufacturingStatus;
|
||||||
|
use App\Services\InfoProviderSystem\DTOs\FileDTO;
|
||||||
|
use App\Services\InfoProviderSystem\DTOs\PartDetailDTO;
|
||||||
|
use App\Services\OAuth\OAuthTokenManager;
|
||||||
|
use Symfony\Component\HttpClient\HttpOptions;
|
||||||
|
use Symfony\Component\HttpClient\NativeHttpClient;
|
||||||
|
use Symfony\Contracts\HttpClient\HttpClientInterface;
|
||||||
|
|
||||||
|
class OctopartProvider implements InfoProviderInterface
|
||||||
|
{
|
||||||
|
private const OAUTH_APP_NAME = 'ip_octopart_oauth';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This defines what fields are returned in the answer from the Octopart API
|
||||||
|
*/
|
||||||
|
private const GRAPHQL_PART_SECTION = <<<'GRAPHQL'
|
||||||
|
{
|
||||||
|
id
|
||||||
|
mpn
|
||||||
|
octopartUrl
|
||||||
|
manufacturer {
|
||||||
|
name
|
||||||
|
}
|
||||||
|
shortDescription
|
||||||
|
category {
|
||||||
|
name
|
||||||
|
path
|
||||||
|
}
|
||||||
|
bestImage {
|
||||||
|
url
|
||||||
|
}
|
||||||
|
bestDatasheet {
|
||||||
|
url
|
||||||
|
name
|
||||||
|
}
|
||||||
|
extras {
|
||||||
|
lifeCycle
|
||||||
|
}
|
||||||
|
manufacturerUrl
|
||||||
|
medianPrice1000 {
|
||||||
|
price
|
||||||
|
currency
|
||||||
|
quantity
|
||||||
|
}
|
||||||
|
sellers(authorizedOnly: false) {
|
||||||
|
company {
|
||||||
|
name
|
||||||
|
homepageUrl
|
||||||
|
}
|
||||||
|
isAuthorized
|
||||||
|
offers {
|
||||||
|
clickUrl
|
||||||
|
inventoryLevel
|
||||||
|
moq
|
||||||
|
packaging
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
GRAPHQL;
|
||||||
|
|
||||||
|
|
||||||
|
public function __construct(private readonly HttpClientInterface $httpClient,
|
||||||
|
private readonly OAuthTokenManager $authTokenManager)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the latest OAuth token for the Octopart API, or creates a new one if none is available
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private function getToken(): string
|
||||||
|
{
|
||||||
|
//Check if we already have a token saved for this app, otherwise we have to retrieve one via OAuth
|
||||||
|
if (!$this->authTokenManager->hasToken(self::OAUTH_APP_NAME)) {
|
||||||
|
$this->authTokenManager->retrieveClientCredentialsToken(self::OAUTH_APP_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
$tmp = $this->authTokenManager->getAlwaysValidTokenString(self::OAUTH_APP_NAME);
|
||||||
|
if ($tmp === null) {
|
||||||
|
throw new \RuntimeException('Could not retrieve OAuth token for Octopart');
|
||||||
|
}
|
||||||
|
|
||||||
|
return $tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make a GraphQL call to the Octopart API
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
private function makeGraphQLCall(string $query, ?array $variables = null): array
|
||||||
|
{
|
||||||
|
if ($variables === []) {
|
||||||
|
$variables = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$options = (new HttpOptions())
|
||||||
|
->setJson(['query' => $query, 'variables' => $variables])
|
||||||
|
->setAuthBearer($this->getToken())
|
||||||
|
;
|
||||||
|
|
||||||
|
$response = $this->httpClient->request(
|
||||||
|
'POST',
|
||||||
|
'https://api.nexar.com/graphql/',
|
||||||
|
$options->toArray(),
|
||||||
|
);
|
||||||
|
|
||||||
|
return $response->toArray(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getProviderInfo(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'name' => 'Octopart',
|
||||||
|
'description' => 'This provider uses the Nexar/Octopart API to search for parts on Octopart.',
|
||||||
|
'url' => 'https://www.octopart.com/',
|
||||||
|
'disabled_help' => 'Set the PROVIDER_OCTOPART_CLIENT_ID and PROVIDER_OCTOPART_SECRET env option.'
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getProviderKey(): string
|
||||||
|
{
|
||||||
|
return 'octopart';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isActive(): bool
|
||||||
|
{
|
||||||
|
//The client ID has to be set and a token has to be available (user clicked connect)
|
||||||
|
//return /*!empty($this->clientId) && */ $this->authTokenManager->hasToken(self::OAUTH_APP_NAME);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function partResultToDTO(array $part): PartDetailDTO
|
||||||
|
{
|
||||||
|
return new PartDetailDTO(
|
||||||
|
provider_key: $this->getProviderKey(),
|
||||||
|
provider_id: $part['id'],
|
||||||
|
name: $part['mpn'],
|
||||||
|
description: $part['shortDescription'],
|
||||||
|
category: $part['category']['name'],
|
||||||
|
manufacturer: $part['manufacturer']['name'],
|
||||||
|
mpn: $part['mpn'],
|
||||||
|
preview_image_url: $part['bestImage']['url'],
|
||||||
|
manufacturing_status: ManufacturingStatus::NOT_SET, //TODO
|
||||||
|
provider_url: $part['octopartUrl'],
|
||||||
|
datasheets: [new FileDTO($part['bestDatasheet']['url'], $part['bestDatasheet']['name'])],
|
||||||
|
manufacturer_product_url: $part['manufacturerUrl'], //TODO
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function searchByKeyword(string $keyword): array
|
||||||
|
{
|
||||||
|
$graphQL = sprintf(<<<'GRAPHQL'
|
||||||
|
query partSearch($keyword: String, $limit: Int) {
|
||||||
|
supSearch(
|
||||||
|
q: $keyword
|
||||||
|
inStockOnly: false
|
||||||
|
limit: $limit
|
||||||
|
) {
|
||||||
|
hits
|
||||||
|
results {
|
||||||
|
part
|
||||||
|
%s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
GRAPHQL, self::GRAPHQL_PART_SECTION);
|
||||||
|
|
||||||
|
|
||||||
|
$result = $this->makeGraphQLCall($graphQL, [
|
||||||
|
'keyword' => $keyword,
|
||||||
|
'limit' => 4,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$tmp = [];
|
||||||
|
|
||||||
|
foreach ($result['data']['supSearch']['results'] as $p) {
|
||||||
|
$tmp[] = $this->partResultToDTO($p['part']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public function getDetails(string $id): PartDetailDTO
|
||||||
|
{
|
||||||
|
$graphql = sprintf(<<<'GRAPHQL'
|
||||||
|
query partSearch($ids: [String!]!) {
|
||||||
|
supParts(ids: $ids)
|
||||||
|
%s
|
||||||
|
}
|
||||||
|
GRAPHQL, self::GRAPHQL_PART_SECTION);
|
||||||
|
|
||||||
|
dump($graphql);
|
||||||
|
|
||||||
|
$result = $this->makeGraphQLCall($graphql, [
|
||||||
|
'ids' => [$id],
|
||||||
|
]);
|
||||||
|
|
||||||
|
return $this->partResultToDTO($result['data']['supParts'][0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCapabilities(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
ProviderCapabilities::BASIC,
|
||||||
|
ProviderCapabilities::FOOTPRINT,
|
||||||
|
ProviderCapabilities::PICTURE,
|
||||||
|
ProviderCapabilities::DATASHEET,
|
||||||
|
ProviderCapabilities::PRICE,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue