Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactoring improvements housekeeping #34

Merged
merged 46 commits into from
Aug 21, 2019
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
46 commits
Select commit Hold shift + click to select a range
3e1858e
Removed php-cs-fixer - we will replace with the much stricter doctrin…
asgrim Aug 20, 2019
6371abb
Added doctrine/coding-standard to apply stricter coding standards acr…
asgrim Aug 20, 2019
fc6297c
Apply new coding standards across codebase
asgrim Aug 20, 2019
47ceebf
Added Psalm to run static analysis on codebase
asgrim Aug 20, 2019
4ffb5f9
Fix issues found by static analysis
asgrim Aug 20, 2019
3a976d7
Fixing up unit test execution so it passes again
asgrim Aug 20, 2019
085069d
Updated documentation with new tools and formatting consistency
asgrim Aug 20, 2019
d5fe5d4
Define interface for type coercion and move to separate namespace
asgrim Aug 20, 2019
d10befb
Move config sources into logical namespace
asgrim Aug 20, 2019
f47afb5
Add missing @covers annotations and static method usage
asgrim Aug 20, 2019
192f0e5
Prefer type-safe assertions in test suite
asgrim Aug 20, 2019
fc14fa4
Fixing new CS issues arising from refactor
asgrim Aug 20, 2019
3570ffd
Move classes related to CoreAgent into separate namespace and made @i…
asgrim Aug 20, 2019
c1dea25
Audit classes for @internal and @deprecated markers
asgrim Aug 20, 2019
5eb2c7b
Add interface for Connector
asgrim Aug 20, 2019
763a5bb
Use constructor injection for creating an Agent
asgrim Aug 20, 2019
d878ca0
Deprecate Agent#getRequest()
asgrim Aug 20, 2019
2287d37
Eliminated multiple unused classes
asgrim Aug 20, 2019
6e779e4
Move Tag, Span, and Request into namespaces
asgrim Aug 20, 2019
af83263
Replace usage of Uuids with semantic RequestId and SpanId which give …
asgrim Aug 20, 2019
f81a6ca
Integration tests should not generate coverage
asgrim Aug 20, 2019
b9e9b19
Serialize IDs properly
asgrim Aug 20, 2019
e125940
Propose better way of injecting configuration to Agent - requires a r…
asgrim Aug 20, 2019
e36e881
Fixes to DerivedSource to use correct keys and also stop using magic …
asgrim Aug 20, 2019
382bf0b
Create injector from config externally to invert dependency on Config
asgrim Aug 20, 2019
760366e
Ignored Endpoints to use a string array instead of Config directly
asgrim Aug 20, 2019
80d59e4
Inject download URL to CoreAgent\Downloader
asgrim Aug 20, 2019
5d2b8d3
Fix Law of Demeter and Liskov substitution issues with CoreAgent\Manager
asgrim Aug 20, 2019
1ece5aa
Eliminate getConfig
asgrim Aug 20, 2019
32a0e05
Removed setConfig method
asgrim Aug 20, 2019
a448dbb
Removed get/setLogger methods from Agent
asgrim Aug 20, 2019
70cc3f9
Make things private in CoreAgent where they don't need to be public
asgrim Aug 20, 2019
b3b3ea9
Fix up namespacing and naming of events
asgrim Aug 20, 2019
3f663e5
Added todo for API discussion
asgrim Aug 20, 2019
b0fccdd
Fixed cs for @todo added
asgrim Aug 20, 2019
52bf608
Updated to make Connector accept a SerializableMessage and shift resp…
asgrim Aug 21, 2019
aebfcc3
Added comment about Tag* class naming
asgrim Aug 21, 2019
b168af1
Capture messages in a delegator in the Agent integration test
asgrim Aug 21, 2019
f5304e9
Added basic docs on using the base library
asgrim Aug 21, 2019
d68befc
Added comment back in regarding having to sleep
asgrim Aug 21, 2019
2eca2d0
Fixing cs and analysis issues
asgrim Aug 21, 2019
ea7c2c3
Set a default for ignore config key to avoid null check
asgrim Aug 21, 2019
ac5effa
Removed commented lines of code about unused property
asgrim Aug 21, 2019
6866b1e
Run phpcbf to fix CS issue
asgrim Aug 21, 2019
9ecc85f
RegisterMessage should be @internal
asgrim Aug 21, 2019
ee7df77
Renamed SerializableMessage to Command to use consistent terminology …
asgrim Aug 21, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 20 additions & 8 deletions src/Agent.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
namespace Scoutapm;

use Closure;
use DateTimeImmutable;
use DateTimeZone;
use Exception;
use Psr\Log\LoggerInterface;
use Psr\Log\NullLogger;
Expand All @@ -13,6 +15,8 @@
use Scoutapm\Connector\SocketConnector;
use Scoutapm\CoreAgent\AutomaticDownloadAndLaunchManager;
use Scoutapm\CoreAgent\Downloader;
use Scoutapm\Events\Metadata;
use Scoutapm\Events\RegisterMessage;
use Scoutapm\Events\Request\Request;
use Scoutapm\Events\Request\RequestId;
use Scoutapm\Events\Span\Span;
Expand Down Expand Up @@ -63,12 +67,7 @@ public function __construct(Config $configuration, Connector $connector, LoggerI

private static function createConnectorFromConfig(Config $config) : SocketConnector
{
return new SocketConnector(
$config->get('socket_path'),
(string) $config->get('name'),
(string) $config->get('key'),
$config->get('api_version')
);
return new SocketConnector($config->get('socket_path'));
}

/**
Expand Down Expand Up @@ -237,8 +236,21 @@ public function send() : bool
return false;
}

// Send this request off to the CoreAgent
return $this->connector->sendRequest($this->request);
if (!$this->connector->sendMessage(new RegisterMessage(
(string) $this->config->get('name'),
(string) $this->config->get('key'),
$this->config->get('api_version')
))) {
return false;
}

if (!$this->connector->sendMessage(new Metadata(
new DateTimeImmutable('now', new DateTimeZone('UTC'))
))) {
return false;
}

return $this->connector->sendMessage($this->request);
}

/**
Expand Down
12 changes: 12 additions & 0 deletions src/Config.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,18 @@ public function __construct()
];
}

/** @param mixed[]|array<string, mixed> $config */
public static function fromArray(array $config = []) : self
{
$instance = new self();

foreach ($config as $key => $value) {
$instance->set($key, $value);
}

return $instance;
}

/**
* Looks through all available sources for the first that can handle this
* key, then returns the value from that source.
Expand Down
7 changes: 5 additions & 2 deletions src/Connector/Connector.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,15 +4,18 @@

namespace Scoutapm\Connector;

use Scoutapm\Events\Request\Request;
use Scoutapm\Connector\Exception\FailedToConnect;
use Scoutapm\Connector\Exception\NotConnected;

interface Connector
{
/** @throws FailedToConnect */
public function connect() : void;

public function connected() : bool;

public function sendRequest(Request $request) : bool;
/** @throws NotConnected */
public function sendMessage(SerializableMessage $message) : bool;

public function shutdown() : void;
}
19 changes: 19 additions & 0 deletions src/Connector/Exception/FailedToConnect.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
<?php

declare(strict_types=1);

namespace Scoutapm\Connector\Exception;

use RuntimeException;

final class FailedToConnect extends RuntimeException
{
public static function fromSocketPathAndPrevious(string $socketPath, \Throwable $previous) : self
{
return new self(sprintf(
'Failed to connect to socket on path "%s", previous message: %s',
$socketPath,
$previous->getMessage()
));
}
}
18 changes: 18 additions & 0 deletions src/Connector/Exception/NotConnected.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
<?php

declare(strict_types=1);

namespace Scoutapm\Connector\Exception;

use RuntimeException;

final class NotConnected extends RuntimeException
{
public static function fromSocketPath(string $socketPath) : self
{
return new self(sprintf(
'Connector has not been connected to socket path "%s"',
$socketPath
));
}
}
11 changes: 11 additions & 0 deletions src/Connector/SerializableMessage.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<?php

declare(strict_types=1);

namespace Scoutapm\Connector;

use JsonSerializable;

interface SerializableMessage extends JsonSerializable
asgrim marked this conversation as resolved.
Show resolved Hide resolved
{
}
70 changes: 23 additions & 47 deletions src/Connector/SocketConnector.php
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,8 @@

namespace Scoutapm\Connector;

use DateTimeImmutable;
use DateTimeZone;
use Exception;
use Scoutapm\Events\Metadata;
use Scoutapm\Events\Request\Request;
use Scoutapm\Connector\Exception\FailedToConnect;
use Scoutapm\Connector\Exception\NotConnected;
use Throwable;
use const AF_UNIX;
use const SOCK_STREAM;
Expand Down Expand Up @@ -37,33 +34,32 @@ final class SocketConnector implements Connector
/** @var string */
private $socketPath;

/** @var string */
private $appName;

/** @var string */
private $appKey;

/** @var string */
private $apiVersion;

public function __construct(string $socketPath, string $appName, string $appKey, string $apiVersion)
public function __construct(string $socketPath)
{
$this->socketPath = $socketPath;
$this->appName = $appName;
$this->appKey = $appKey;
$this->apiVersion = $apiVersion;

$this->socket = socket_create(AF_UNIX, SOCK_STREAM, 0);
$this->connect();
register_shutdown_function([&$this, 'shutdown']);

// Pre-emptive attempt to connect, strictly speaking `__construct` should not have side-effects, so if this
// fails then swallow it. The `Agent` goes on to call connect() anyway, and handles launching of the core agent.
try {
$this->connect();
} catch (FailedToConnect $failedToConnect) {
}
}

public function connect() : void
{
if ($this->connected()) {
return;
}

try {
$this->connected = socket_connect($this->socket, $this->socketPath);
register_shutdown_function([&$this, 'shutdown']);
} catch (Throwable $e) {
$this->connected = false;
throw FailedToConnect::fromSocketPathAndPrevious($this->socketPath, $e);
}
}

Expand All @@ -72,11 +68,12 @@ public function connected() : bool
return $this->connected;
}

/**
* @param mixed $message
*/
private function sendMessage($message) : void
public function sendMessage(SerializableMessage $message) : bool
{
if (!$this->connected()) {
throw NotConnected::fromSocketPath($this->socketPath);
}

$serializedJsonString = json_encode($message);

$size = strlen($serializedJsonString);
Expand All @@ -85,36 +82,15 @@ private function sendMessage($message) : void

// Read the response back and drop it. Needed for socket liveness
$responseLength = socket_read($this->socket, 4);
/** @noinspection UnusedFunctionResultInspection */
socket_read($this->socket, unpack('N', $responseLength)[1]);
}

/** @throws Exception */
public function sendRequest(Request $request) : bool
{
$this->sendMessage([
'Register' => [
'app' => $this->appName,
'key' => $this->appKey,
'language' => 'php',
'api_version' => $this->apiVersion,
],
]);

$this->sendMessage(new Metadata(
new DateTimeImmutable('now', new DateTimeZone('UTC'))
));

// Send the whole Request as a batch command
$this->sendMessage([
'BatchCommand' => ['commands' => $request],
]);

return socket_last_error($this->socket) === 0;
}

public function shutdown() : void
{
if ($this->connected !== true) {
if (!$this->connected()) {
return;
}

Expand Down
4 changes: 2 additions & 2 deletions src/Events/Metadata.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
namespace Scoutapm\Events;

use DateTimeImmutable;
use JsonSerializable;
use Scoutapm\Connector\SerializableMessage;
use Scoutapm\Helper\Timer;
use const PHP_VERSION;
use function gethostname;
Expand All @@ -15,7 +15,7 @@
*
* @internal
*/
final class Metadata implements JsonSerializable
final class Metadata implements SerializableMessage
{
/** @var Timer */
private $timer;
Expand Down
38 changes: 38 additions & 0 deletions src/Events/RegisterMessage.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
<?php

declare(strict_types=1);

namespace Scoutapm\Events;

use Scoutapm\Connector\SerializableMessage;

final class RegisterMessage implements SerializableMessage
{
/** @var string */
private $appName;

/** @var string */
private $appKey;

/** @var string */
private $apiVersion;

public function __construct(string $appName, string $appKey, string $apiVersion)
{
$this->appName = $appName;
$this->appKey = $appKey;
$this->apiVersion = $apiVersion;
}

public function jsonSerialize() : array
{
return [
'Register' => [
'app' => $this->appName,
'key' => $this->appKey,
'language' => 'php',
'api_version' => $this->apiVersion,
],
];
}
}
12 changes: 6 additions & 6 deletions src/Events/Request/Request.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
namespace Scoutapm\Events\Request;

use Exception;
use JsonSerializable;
use Scoutapm\Connector\SerializableMessage;
use Scoutapm\Events\Exception\SpanHasNotBeenStarted;
use Scoutapm\Events\Span\Span;
use Scoutapm\Events\Tag\TagRequest;
Expand All @@ -16,7 +16,7 @@
use function end;

/** @internal */
class Request implements JsonSerializable
class Request implements SerializableMessage
{
/** @var Timer */
private $timer;
Expand Down Expand Up @@ -110,9 +110,7 @@ public function jsonSerialize() : array
];

foreach ($this->events as $event) {
$array = $event->jsonSerialize();

foreach ($array as $value) {
foreach ($event->jsonSerialize() as $value) {
$commands[] = $value;
}
}
Expand All @@ -124,7 +122,9 @@ public function jsonSerialize() : array
],
];

return $commands;
return [
'BatchCommand' => ['commands' => $commands],
];
}

/**
Expand Down
4 changes: 2 additions & 2 deletions src/Events/Span/Span.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,13 @@
namespace Scoutapm\Events\Span;

use Exception;
use JsonSerializable;
use Scoutapm\Connector\SerializableMessage;
use Scoutapm\Events\Request\RequestId;
use Scoutapm\Events\Tag\TagSpan;
use Scoutapm\Helper\Timer;

/** @internal */
class Span implements JsonSerializable
class Span implements SerializableMessage
{
/** @var SpanId */
private $id;
Expand Down
Loading