templates/features/integrations.twig line 1

Open in your IDE?
  1. {% set title = 'Our 3rd Party Integrations and Support' %}
  2. {% set metaDescription = 'Invoiced integrates with a range of accounting software solutions, and we are growing our list of integration partners by the day.' %}
  3. {% set canonicalUrl = url('product_integrations') %}
  4. {% set htmlClass = 'gray-bg' %}
  5. {% set navbarClass = 'navbar-gray' %}
  6. {% extends "parent.twig" %}
  7. {% block content %}
  8.     <section class="content features integrations">
  9.         <div class="container">
  10.             <h1 class="title">Integrations</h1>
  11.             <h2 class="subtitle">
  12.                 Invoiced works with these third-party systems out of the box.
  13.             </h2>
  14.         </div>
  15.         <div class="supported-integrations">
  16.             <div class="container">
  17.                 <h3>Accounting / ERP Integrations</h3>
  18.                 <div class="integration-tiles clearfix">
  19.                     <a href="{{ path('product_erp_connect') }}" class="tile integration">
  20.                         <div class="logo">
  21.                             <img src="{{ asset('/img/integrations/erp-connect.png') }}" alt="ERP Connect"/>
  22.                         </div>
  23.                     </a>
  24.                     <a href="{{ path('netsuite_landing') }}" class="tile integration">
  25.                         <div class="logo">
  26.                             <img src="{{ asset('/img/integrations/netsuite.png') }}" alt="NetSuite"/>
  27.                         </div>
  28.                     </a>
  29.                     <a href="{{ path('quickbooks_landing') }}" class="tile integration">
  30.                         <div class="logo">
  31.                             <img src="{{ asset('/img/integrations/quickbooks-online.png') }}" alt="QuickBooks Online"/>
  32.                         </div>
  33.                     </a>
  34.                     <a href="{{ path('quickbooks_landing') }}" class="tile integration">
  35.                         <div class="logo">
  36.                             <img src="{{ asset('/img/integrations/quickbooks-desktop.png') }}"
  37.                                  alt="QuickBooks Desktop"/>
  38.                         </div>
  39.                     </a>
  40.                     <a href="{{ path('quickbooks_landing') }}" class="tile integration">
  41.                         <div class="logo">
  42.                             <img src="{{ asset('/img/integrations/quickbooks-enterprise.png') }}"
  43.                                  alt="QuickBooks Enterprise"/>
  44.                         </div>
  45.                     </a>
  46.                     <a href="{{ path('intacct_landing') }}" class="tile integration">
  47.                         <div class="logo">
  48.                             <img src="{{ asset('/img/integrations/intacct.png') }}" alt="Sage Intacct"/>
  49.                         </div>
  50.                     </a>
  51.                     <a href="{{ path('xero_landing') }}" class="tile integration">
  52.                         <div class="logo">
  53.                             <img src="{{ asset('/img/integrations/xero.png') }}" alt="Xero"/>
  54.                         </div>
  55.                     </a>
  56.                 </div>
  57.                 <h3>Specialty Integrations</h3>
  58.                 <div class="integration-tiles clearfix">
  59.                     <a href="https://docs.invoiced.com//integrations/avalara" class="tile integration">
  60.                         <div class="logo">
  61.                             <img src="{{ asset('/img/integrations/avalara.png') }}" alt="Avalara"/>
  62.                         </div>
  63.                     </a>
  64.                     <div class="tile integration">
  65.                         <div class="logo">
  66.                             <img src="{{ asset('/img/integrations/automationsio.png') }}" alt="Automations.Io"/>
  67.                         </div>
  68.                     </div>
  69.                     <a href="https://docs.invoiced.com//integrations/chartmogul" class="tile integration">
  70.                         <div class="logo">
  71.                             <img src="{{ asset('/img/integrations/chartmogul.png') }}" alt="ChartMogul"/>
  72.                         </div>
  73.                     </a>
  74.                     <div class="tile integration">
  75.                         <div class="logo">
  76.                             <img src="{{ asset('/img/integrations/collbox.png') }}" alt="Collbox"/>
  77.                         </div>
  78.                     </div>
  79.                     <a href="https://docs.invoiced.com//integrations/earth-class-mail"
  80.                        class="tile integration">
  81.                         <div class="logo">
  82.                             <img src="{{ asset('/img/integrations/earthclassmail.svg') }}" alt="Earth Class Mail"/>
  83.                         </div>
  84.                     </a>
  85.                     <a href="https://docs.invoiced.com//integrations/lob" class="tile integration">
  86.                         <div class="logo">
  87.                             <img src="{{ asset('/img/integrations/lob.png') }}" alt="Lob"/>
  88.                         </div>
  89.                     </a>
  90.                     <div class="tile integration">
  91.                         <div class="logo">
  92.                             <img src="{{ asset('/img/integrations/missing-link-technology.png') }}"
  93.                                  alt="Missing Link Technology"/>
  94.                         </div>
  95.                     </div>
  96.                     <a href="https://docs.invoiced.com//integrations/salesforce" class="tile integration">
  97.                         <div class="logo">
  98.                             <img src="{{ asset('/img/integrations/salesforce.png') }}" alt="Salesforce"/>
  99.                         </div>
  100.                     </a>
  101.                     <a href="https://docs.invoiced.com//integrations/slack" class="tile integration">
  102.                         <div class="logo">
  103.                             <img src="{{ asset('/img/integrations/slack.png') }}" alt="Slack"/>
  104.                         </div>
  105.                     </a>
  106.                     <div class="tile integration">
  107.                         <div class="logo">
  108.                             <img src="{{ asset('/img/integrations/stitch.png') }}" alt="Stitch"/>
  109.                         </div>
  110.                     </div>
  111.                     <a href="https://docs.invoiced.com//integrations/twilio" class="tile integration">
  112.                         <div class="logo">
  113.                             <img src="{{ asset('/img/integrations/twilio.png') }}" alt="Twilio"/>
  114.                         </div>
  115.                     </a>
  116.                     <a href="https://docs.invoiced.com//integrations/zapier" class="tile integration">
  117.                         <div class="logo">
  118.                             <img src="{{ asset('/img/integrations/zapier.png') }}" alt="Zapier"/>
  119.                         </div>
  120.                     </a>
  121.                 </div>
  122.                 <a name="payment-gateways"></a>
  123.                 <h3>Payment Gateways</h3>
  124.                 <div class="integration-tiles clearfix">
  125.                     <div class="tile integration">
  126.                         <div class="preferred"><span class="label label-success">Preferred</span></div>
  127.                         <div class="logo">
  128.                             <img src="{{ asset('/img/gateways/stripe.png') }}" alt="Stripe"/>
  129.                         </div>
  130.                     </div>
  131.                     <div class="tile integration">
  132.                         <div class="logo">
  133.                             <img src="{{ asset('/img/gateways/amex.png') }}" alt="American Express"/>
  134.                         </div>
  135.                     </div>
  136.                     <div class="tile integration">
  137.                         <div class="logo">
  138.                             <img src="{{ asset('/img/gateways/authorizenet.png') }}" alt="Authorize.Net"/>
  139.                         </div>
  140.                     </div>
  141.                     <div class="tile integration">
  142.                         <div class="logo">
  143.                             <img src="{{ asset('/img/gateways/braintree.png') }}" alt="Braintree"/>
  144.                         </div>
  145.                     </div>
  146.                     <div class="tile integration">
  147.                         <div class="logo">
  148.                             <img src="{{ asset('/img/gateways/chase_paymentech.png') }}" alt="Chase Paymentech Orbital"/>
  149.                         </div>
  150.                     </div>
  151.                     <div class="tile integration">
  152.                         <div class="logo">
  153.                             <img src="{{ asset('/img/gateways/cardknox.png') }}" alt="CardKnox"/>
  154.                         </div>
  155.                     </div>
  156.                     <div class="tile integration">
  157.                         <div class="logo">
  158.                             <img src="{{ asset('/img/gateways/cybersource.png') }}" alt="CyberSource"/>
  159.                         </div>
  160.                     </div>
  161.                     <div class="tile integration">
  162.                         <div class="logo">
  163.                             <img src="{{ asset('/img/gateways/gocardless.png') }}" alt="GoCardless"/>
  164.                         </div>
  165.                     </div>
  166.                     <div class="tile integration">
  167.                         <div class="logo">
  168.                             <img src="{{ asset('/img/gateways/nmi.png') }}" alt="NMI"/>
  169.                         </div>
  170.                     </div>
  171.                     <div class="tile integration">
  172.                         <div class="logo">
  173.                             <img src="{{ asset('/img/gateways/paypal_express_checkout.png') }}" alt="PayPal Express Checkout"/>
  174.                         </div>
  175.                     </div>
  176.                     <div class="tile integration">
  177.                         <div class="logo">
  178.                             <img src="{{ asset('/img/gateways/paypal_payflow_pro.png') }}" alt="PayPal Payflow Pro"/>
  179.                         </div>
  180.                     </div>
  181.                 </div>
  182.                 <div class="missing-integration">
  183.                     <h3>Don't see your integration listed?</h3>
  184.                     <p>
  185.                         With our <a href="https://docs.invoiced.com/dev">API</a>, Invoiced can be
  186.                         integrated with any software system, including accounting, CRMs, ordering, business
  187.                         intelligence, and other backoffice systems.
  188.                     </p>
  189.                 </div>
  190.             </div>
  191.         </div>
  192.     </section>
  193. {% endblock %}
HTTP/2 401 returned for "https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240". (500 Internal Server Error)

Symfony Exception

ClientException

HTTP 500 Internal Server Error

HTTP/2 401 returned for "https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240".

Exception

Symfony\Component\HttpClient\Exception\ ClientException

  1.         if (500 <= $code) {
  2.             throw new ServerException($this);
  3.         }
  4.         if (400 <= $code) {
  5.             throw new ClientException($this);
  6.         }
  7.         if (300 <= $code) {
  8.             throw new RedirectionException($this);
  9.         }
  1.         } finally {
  2.             if ($this->event && $this->event->isStarted()) {
  3.                 $this->event->stop();
  4.             }
  5.             if ($throw) {
  6.                 $this->checkStatusCode($this->response->getStatusCode());
  7.             }
  8.         }
  9.     }
  10.     public function toArray(bool $throw true): array
TraceableResponse->getContent() in src/Blog.php (line 181)
  1.                     'hapikey' => $this->hubSpotKey,
  2.                     'after' => $after,
  3.                 ],
  4.             ]);
  5.             $result json_decode($response->getContent(), true);
  6.         } catch (HttpExceptionInterface $e) {
  7.             $this->logger->error('Could not load blog posts from HubSpot', ['exception' => $e]);
  8.             throw $e;
  9.         }
Blog->loadBlogAuthors() in src/Blog.php (line 152)
  1.         if (!isset($this->authors)) {
  2.             $beta $bustCache INF null;
  3.             $this->authors $this->cache->get('blog_authors', function (ItemInterface $item) {
  4.                 $item->expiresAfter(86400); // cache for 1 day
  5.                 return $this->loadBlogAuthors();
  6.             }, $beta);
  7.         }
  8.         return $this->authors;
  9.     }
  1.         $isHit true;
  2.         $callback = function (CacheItem $itembool &$save) use ($callback, &$isHit) {
  3.             $isHit $item->isHit();
  4.             return $callback($item$save);
  5.         };
  6.         $event $this->start(__FUNCTION__);
  7.         try {
  8.             $value $this->pool->get($key$callback$beta$metadata);
in vendor/symfony/cache/LockRegistry.php -> Symfony\Component\Cache\Adapter\{closure} (line 108)
  1.                 if ($locked || !$wouldBlock) {
  2.                     $logger && $logger->info(sprintf('Lock %s, now computing item "{key}"'$locked 'acquired' 'not supported'), ['key' => $item->getKey()]);
  3.                     self::$lockedFiles[$key] = true;
  4.                     $value $callback($item$save);
  5.                     if ($save) {
  6.                         if ($setMetadata) {
  7.                             $setMetadata($item);
  8.                         }
  1.             }
  2.             try {
  3.                 $value = ($this->callbackWrapper)($callback$item$save$pool, function (CacheItem $item) use ($setMetadata$startTime, &$metadata) {
  4.                     $setMetadata($item$startTime$metadata);
  5.                 }, $this->logger ?? null);
  6.                 $setMetadata($item$startTime$metadata);
  7.                 return $value;
  8.             } finally {
  9.                 unset($this->computing[$key]);
in vendor/symfony/cache-contracts/CacheTrait.php -> Symfony\Component\Cache\Traits\{closure} (line 72)
  1.             }
  2.         }
  3.         if ($recompute) {
  4.             $save true;
  5.             $item->set($callback($item$save));
  6.             if ($save) {
  7.                 $pool->save($item);
  8.             }
  9.         }
  1.                 return $value;
  2.             } finally {
  3.                 unset($this->computing[$key]);
  4.             }
  5.         }, $beta$metadata$this->logger ?? null);
  6.     }
  7. }
  1.      *
  2.      * @return mixed
  3.      */
  4.     public function get(string $key, callable $callbackfloat $beta null, array &$metadata null)
  5.     {
  6.         return $this->doGet($this$key$callback$beta$metadata);
  7.     }
  8.     /**
  9.      * {@inheritdoc}
  10.      */
  1.             return $callback($item$save);
  2.         };
  3.         $event $this->start(__FUNCTION__);
  4.         try {
  5.             $value $this->pool->get($key$callback$beta$metadata);
  6.             $event->result[$key] = get_debug_type($value);
  7.         } finally {
  8.             $event->end microtime(true);
  9.         }
  10.         if ($isHit) {
TraceableAdapter->get('blog_authors', object(Closure), INF) in src/Blog.php (line 149)
  1.     public function getAuthors(bool $bustCache false): array
  2.     {
  3.         if (!isset($this->authors)) {
  4.             $beta $bustCache INF null;
  5.             $this->authors $this->cache->get('blog_authors', function (ItemInterface $item) {
  6.                 $item->expiresAfter(86400); // cache for 1 day
  7.                 return $this->loadBlogAuthors();
  8.             }, $beta);
  9.         }
Blog->getAuthors(true) in src/Blog.php (line 37)
  1.      * WARNING: This will be slow and should not be called
  2.      * unless intentional.
  3.      */
  4.     public function bustCache(): void
  5.     {
  6.         $this->getAuthors(true);
  7.         $this->getTags(true);
  8.         $this->getAllPosts(true);
  9.     }
  10.     public function getAuthor(string $slug): ?array
  1.         // Update the last refresh time now to minimize cache stampede
  2.         $this->cache->get('lastContentRefresh', function () {
  3.             return time();
  4.         }, INF);
  5.         $this->blog->bustCache();
  6.     }
  7.     public static function getSubscribedEvents(): array
  8.     {
  9.         return [
ContentRefreshSubscriber->refresh() in src/EventSubscriber/ContentRefreshSubscriber.php (line 34)
  1.         $lastRefresh $this->cache->get('lastContentRefresh', function () {
  2.             return 0;
  3.         });
  4.         if ($lastRefresh strtotime('-5 minutes')) {
  5.             $this->refresh();
  6.         }
  7.     }
  8.     private function refresh(): void
  9.     {
  1.                     $closure = static function (...$args) use (&$listener, &$closure) {
  2.                         if ($listener[0] instanceof \Closure) {
  3.                             $listener[0] = $listener[0]();
  4.                             $listener[1] = $listener[1] ?? '__invoke';
  5.                         }
  6.                         ($closure \Closure::fromCallable($listener))(...$args);
  7.                     };
  8.                 } else {
  9.                     $closure $listener instanceof \Closure || $listener instanceof WrappedListener $listener \Closure::fromCallable($listener);
  10.                 }
  11.             }
in vendor/symfony/event-dispatcher/EventDispatcher.php :: Symfony\Component\EventDispatcher\{closure} (line 230)
  1.         foreach ($listeners as $listener) {
  2.             if ($stoppable && $event->isPropagationStopped()) {
  3.                 break;
  4.             }
  5.             $listener($event$eventName$this);
  6.         }
  7.     }
  8.     /**
  9.      * Sorts the internal list of listeners for the given event by priority.
  1.         } else {
  2.             $listeners $this->getListeners($eventName);
  3.         }
  4.         if ($listeners) {
  5.             $this->callListeners($listeners$eventName$event);
  6.         }
  7.         return $event;
  8.     }
  1.     /**
  2.      * {@inheritdoc}
  3.      */
  4.     public function terminate(Request $requestResponse $response)
  5.     {
  6.         $this->dispatcher->dispatch(new TerminateEvent($this$request$response), KernelEvents::TERMINATE);
  7.     }
  8.     /**
  9.      * @internal
  10.      */
in vendor/symfony/http-kernel/Kernel.php -> terminate (line 159)
  1.         if (false === $this->booted) {
  2.             return;
  3.         }
  4.         if ($this->getHttpKernel() instanceof TerminableInterface) {
  5.             $this->getHttpKernel()->terminate($request$response);
  6.         }
  7.     }
  8.     /**
  9.      * {@inheritdoc}
Kernel->terminate(object(Request), object(Response)) in public/index.php (line 36)
  1. $kernel = new Kernel($_SERVER['APP_ENV'], (bool) $_SERVER['APP_DEBUG']);
  2. $request Request::createFromGlobals();
  3. $response $kernel->handle($request);
  4. $response->send();
  5. $kernel->terminate($request$response);

Logs 3

Level Channel Message
INFO 14:27:35 php User Deprecated: Since symfony/framework-bundle 5.3: The "session.storage.native" service is deprecated, use "session.storage.factory.native" instead.
{
    "exception": {}
}
INFO 14:27:35 php User Deprecated: Since symfony/framework-bundle 5.3: The "session.storage.metadata_bag" service is deprecated, create your own "session.storage.factory" instead.
{
    "exception": {}
}
INFO 14:27:35 request Matched route "_profiler_open_file".
{
    "route": "_profiler_open_file",
    "route_parameters": {
        "_route": "_profiler_open_file",
        "_controller": "web_profiler.controller.profiler::openAction"
    },
    "request_uri": "http://invoiced.project-release.info/_profiler/open?file=templates%2Ffeatures%2Fintegrations.twig&line=1",
    "method": "GET"
}
INFO 14:27:36 cache Lock acquired, now computing item "lastContentRefresh"
{
    "key": "lastContentRefresh"
}
INFO 14:27:36 cache Lock acquired, now computing item "blog_authors"
{
    "key": "blog_authors"
}
INFO 14:27:36 http_client Request: "GET https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240"
INFO 14:27:36 http_client Response: "401 https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240"
ERROR 14:27:36 app Could not load blog posts from HubSpot
{
    "exception": {}
}
CRITICAL 14:27:36 php Uncaught Exception: HTTP/2 401 returned for "https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240".
{
    "exception": {}
}
CRITICAL 14:27:36 request Uncaught PHP Exception Symfony\Component\HttpClient\Exception\ClientException: "HTTP/2 401 returned for "https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240"." at /var/www/invoiced/data/www/invoiced.project-release.info/vendor/symfony/http-client/Response/TraceableResponse.php line 212
{
    "exception": {}
}

Stack Trace

ClientException
Symfony\Component\HttpClient\Exception\ClientException:
HTTP/2 401  returned for "https://api.hubapi.com/cms/v3/blogs/authors?hapikey=41d1972b-a2ce-454d-9e57-cd6355e00240".

  at vendor/symfony/http-client/Response/TraceableResponse.php:212
  at Symfony\Component\HttpClient\Response\TraceableResponse->checkStatusCode(401)
     (vendor/symfony/http-client/Response/TraceableResponse.php:103)
  at Symfony\Component\HttpClient\Response\TraceableResponse->getContent()
     (src/Blog.php:181)
  at App\Blog->loadBlogAuthors()
     (src/Blog.php:152)
  at App\Blog->App\{closure}(object(CacheItem), true)
     (vendor/symfony/cache/Adapter/TraceableAdapter.php:51)
  at Symfony\Component\Cache\Adapter\TraceableAdapter->Symfony\Component\Cache\Adapter\{closure}(object(CacheItem), true)
     (vendor/symfony/cache/LockRegistry.php:108)
  at Symfony\Component\Cache\LockRegistry::compute(object(Closure), object(CacheItem), true, object(FilesystemAdapter), object(Closure), object(Logger))
     (vendor/symfony/cache/Traits/ContractsTrait.php:100)
  at Symfony\Component\Cache\Adapter\AbstractAdapter->Symfony\Component\Cache\Traits\{closure}(object(CacheItem), true)
     (vendor/symfony/cache-contracts/CacheTrait.php:72)
  at Symfony\Component\Cache\Adapter\AbstractAdapter->contractsGet(object(FilesystemAdapter), 'blog_authors', object(Closure), INF, array(), object(Logger))
     (vendor/symfony/cache/Traits/ContractsTrait.php:107)
  at Symfony\Component\Cache\Adapter\AbstractAdapter->doGet(object(FilesystemAdapter), 'blog_authors', object(Closure), INF, array())
     (vendor/symfony/cache-contracts/CacheTrait.php:35)
  at Symfony\Component\Cache\Adapter\AbstractAdapter->get('blog_authors', object(Closure), INF, array())
     (vendor/symfony/cache/Adapter/TraceableAdapter.php:56)
  at Symfony\Component\Cache\Adapter\TraceableAdapter->get('blog_authors', object(Closure), INF)
     (src/Blog.php:149)
  at App\Blog->getAuthors(true)
     (src/Blog.php:37)
  at App\Blog->bustCache()
     (src/EventSubscriber/ContentRefreshSubscriber.php:45)
  at App\EventSubscriber\ContentRefreshSubscriber->refresh()
     (src/EventSubscriber/ContentRefreshSubscriber.php:34)
  at App\EventSubscriber\ContentRefreshSubscriber->onKernelTerminate(object(TerminateEvent), 'kernel.terminate', object(EventDispatcher))
     (vendor/symfony/event-dispatcher/EventDispatcher.php:270)
  at Symfony\Component\EventDispatcher\EventDispatcher::Symfony\Component\EventDispatcher\{closure}(object(TerminateEvent), 'kernel.terminate', object(EventDispatcher))
     (vendor/symfony/event-dispatcher/EventDispatcher.php:230)
  at Symfony\Component\EventDispatcher\EventDispatcher->callListeners(array(object(Closure), object(Closure)), 'kernel.terminate', object(TerminateEvent))
     (vendor/symfony/event-dispatcher/EventDispatcher.php:59)
  at Symfony\Component\EventDispatcher\EventDispatcher->dispatch(object(TerminateEvent), 'kernel.terminate')
     (vendor/symfony/http-kernel/HttpKernel.php:94)
  at Symfony\Component\HttpKernel\HttpKernel->terminate(object(Request), object(Response))
     (vendor/symfony/http-kernel/Kernel.php:159)
  at Symfony\Component\HttpKernel\Kernel->terminate(object(Request), object(Response))
     (public/index.php:36)