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

[DO NOT MERGE] Add user event to Laravel #2509

Closed
wants to merge 32 commits into from
Closed
Changes from all commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
9e86f5b
Add user event to Laravel 8
estringana Feb 7, 2024
9f99639
Fix issue on WebFrameworkTestCase
estringana Feb 7, 2024
f7c7463
Fix login events test
estringana Feb 7, 2024
94490b8
Keep web session optional
estringana Feb 7, 2024
be5967c
Add login tests to all laravel versions
estringana Feb 7, 2024
07c78ab
Add login events to Laravel 9x
estringana Feb 8, 2024
8a019ce
Fix default string length for Laravel 9x
estringana Feb 8, 2024
87c3889
Update Laravel 9 snapshots
estringana Feb 8, 2024
132912b
Fix cli laravel 9 tests
estringana Feb 8, 2024
d397bec
Add login events to Laravel 10
estringana Feb 8, 2024
3b0e776
Set default string length for Laravel 10
estringana Feb 8, 2024
9e0e3dd
Fix laravel 10 snapshots
estringana Feb 8, 2024
297311c
Fix Laravel 10 cli tests
estringana Feb 8, 2024
4fa5f55
Set user on Laravel 4
estringana Feb 9, 2024
e3a2c2b
Set user on Laravel 5x
estringana Feb 9, 2024
32a08f0
Remove unnecesary line
estringana Feb 12, 2024
0441715
Create loggin snapshots for Laravel 9
estringana Feb 13, 2024
bfa6192
wip
estringana Feb 13, 2024
dfe4ef9
Update snapshots
estringana Feb 13, 2024
87d5b0f
Update snapshots
estringana Feb 13, 2024
c9469c7
Remove redirect
estringana Feb 13, 2024
5b7943b
Fix redirect issue
estringana Feb 14, 2024
e90aafd
Configure database modes
estringana Feb 14, 2024
ad93ecc
Change testing approach
estringana Feb 14, 2024
33d661e
Update laravel 10 snapshots
estringana Feb 14, 2024
c3e64fa
wip
estringana Feb 14, 2024
2946774
Align tests
estringana Feb 14, 2024
33f2a98
Generate missing snapshots
estringana Feb 14, 2024
9d5de49
Name route in laravel 10
estringana Feb 15, 2024
4d40ccf
Change Test Order and check for connection availability
PROFeNoM Feb 15, 2024
bf37a1e
Misc changes
estringana Feb 15, 2024
c1a2b2d
Amend conflicts
estringana Apr 25, 2024
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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
@@ -60,3 +60,4 @@ package.json
packages.json
package-lock.json
yarn.lock
cookies.txt
59 changes: 59 additions & 0 deletions src/DDTrace/Integrations/Laravel/LaravelIntegration.php
Original file line number Diff line number Diff line change
@@ -400,6 +400,7 @@ function ($This, $scope, $args) use ($integration) {
$metadata['email'] = $args[0]['email'];
}

\DDTrace\set_user($args[0]->getAuthIdentifier(), $metadata);
\datadog\appsec\track_user_login_success_event(
\method_exists($args[0], 'getAuthIdentifier') ? $args[0]->getAuthIdentifier() : '',
$metadata,
@@ -408,6 +409,34 @@ function ($This, $scope, $args) use ($integration) {
}
);

// Used by Laravel < 5.0
\DDTrace\hook_method(
'Illuminate\Auth\Guard',
'user',
null,
function ($This, $scope, $args, $user) use ($integration) {
$authClass = 'Illuminate\Auth\UserInterface';
if (
!isset($user) ||
!$user ||
!($user instanceof $authClass) ||
!\method_exists($user, 'getAuthIdentifier')
) {
return;
}

$metadata = [];
if (isset($user['name'])) {
$metadata['name'] = $user['name'];
}
if (isset($user['email'])) {
$metadata['email'] = $user['email'];
}

\DDTrace\set_user($user->getAuthIdentifier(), $metadata);
}
);

// Used by Laravel < 5.0
\DDTrace\hook_method(
'Illuminate\Auth\Guard',
@@ -443,6 +472,36 @@ function ($This, $scope, $args) use ($integration) {
}
);

// Used by Laravel >= 5.0
\DDTrace\hook_method(
'Illuminate\Auth\Events\Authenticated',
'__construct',
null,
function ($This, $scope, $args) use ($integration) {
$authClass = 'Illuminate\Contracts\Auth\Authenticatable';
if (
!isset($args[1]) ||
!$args[1] ||
!($args[1] instanceof $authClass)
) {
return;
}

$meta = [];
$user = $args[1];
if (isset($user->name)) {
$meta['name'] = $user->name;
}
if (isset($user->email)) {
$meta['email'] = $user->email;
}

if (\method_exists($user, 'getAuthIdentifier')) {
\DDTrace\set_user($user->getAuthIdentifier(), $meta);
}
}
);

return Integration::LOADED;
}

30 changes: 30 additions & 0 deletions tests/Common/WebFrameworkTestCase.php
Original file line number Diff line number Diff line change
@@ -21,12 +21,36 @@ abstract class WebFrameworkTestCase extends IntegrationTestCase
const PORT = 9999;

const ERROR_LOG_NAME = 'phpunit_error.log';
const COOKIE_JAR = 'cookies.txt';

/**
* @var WebServer|null
*/
private static $appServer;
protected $checkWebserverErrors = true;
protected $cookiesFile;
protected $maintainSession = false;

protected function ddSetUp()
{
parent::ddSetUp();
}

protected function enableSession()
{
$this->maintainSession = true;
$this->cookiesFile = realpath(dirname(static::getAppIndexScript())) . '/' . static::COOKIE_JAR;
$f = @fopen($this->cookiesFile, "r+");
if ($f !== false) {
ftruncate($f, 0);
fclose($f);
}
}

protected function disableSession()
{
$this->maintainSession = false;
}

public static function ddSetUpBeforeClass()
{
@@ -215,6 +239,12 @@ protected function sendRequest($method, $url, $headers = [], $body = [], $change
curl_setopt($ch, CURLOPT_RETURNTRANSFER, $options[CURLOPT_RETURNTRANSFER]);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, $options[CURLOPT_FOLLOWLOCATION]);
if ($this->maintainSession) {
curl_setopt($ch, CURLOPT_COOKIEJAR, $this->cookiesFile);
curl_setopt ($ch, CURLOPT_COOKIEFILE, $this->cookiesFile);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HEADER, 1);
}
if ($method === 'POST') {
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_POSTFIELDS, is_array($body) ? json_encode($body) : $body);
7 changes: 0 additions & 7 deletions tests/Frameworks/Laravel/Version_10_x/.env
Original file line number Diff line number Diff line change
@@ -8,13 +8,6 @@ LOG_CHANNEL=stack
LOG_DEPRECATIONS_CHANNEL=null
LOG_LEVEL=debug

DB_CONNECTION=mysql
DB_HOST=127.0.0.1
DB_PORT=3306
DB_DATABASE=laravel
DB_USERNAME=root
DB_PASSWORD=

BROADCAST_DRIVER=log
CACHE_DRIVER=file
FILESYSTEM_DISK=local
7 changes: 0 additions & 7 deletions tests/Frameworks/Laravel/Version_10_x/.env.example
Original file line number Diff line number Diff line change
@@ -8,13 +8,6 @@ LOG_CHANNEL=stack
LOG_DEPRECATIONS_CHANNEL=null
LOG_LEVEL=debug

DB_CONNECTION=mysql
DB_HOST=127.0.0.1
DB_PORT=3306
DB_DATABASE=laravel
DB_USERNAME=root
DB_PASSWORD=

BROADCAST_DRIVER=log
CACHE_DRIVER=file
FILESYSTEM_DISK=local
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Auth\Events\Registered;
use Illuminate\Support\Facades\Hash;
use Illuminate\Http\RedirectResponse;
use App\Providers\RouteServiceProvider;
use App\Models\User;

class LoginTestController extends Controller
{
/**
* @param \Illuminate\Http\Request $request
* @return \Illuminate\Http\Response
*/
public function auth(Request $request)
{
$credentials = [
'email' => $request->get('email'),
'password' => 'password',
];

if (Auth::attempt($credentials)) {
return response('Login successful', 200);
}

return response('Invalid credentials', 403);
}

public function register(Request $request): RedirectResponse
{
$request->validate([
'name' => ['required'],
'email' => ['required'],
'password' => ['required'],
]);

$user = User::create([
'name' => $request->name,
'email' => $request->email,
'password' => Hash::make($request->password),
]);

event(new Registered($user));

Auth::login($user);

return redirect('/simple');
}

public function behind_auth()
{
return "page behind auth";
}
}
Original file line number Diff line number Diff line change
@@ -2,6 +2,7 @@

namespace App\Providers;

use Illuminate\Support\Facades\Schema;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
@@ -19,6 +20,6 @@ public function register(): void
*/
public function boot(): void
{
//
Schema::defaultStringLength(191);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just for my understanding and future versions of the framework: What does this have to be set?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's not common in Laravel having to set this odd thing :( but this is the reason https://laravel.com/docs/master/migrations#index-lengths-mysql-mariadb

}
}
6 changes: 4 additions & 2 deletions tests/Frameworks/Laravel/Version_10_x/composer.json
Original file line number Diff line number Diff line change
@@ -30,12 +30,14 @@
"autoload-dev": {
"psr-4": {
"Tests\\": "tests/"
}
},
"files": ["../../../Appsec/Mock.php"]
},
"scripts": {
"post-autoload-dump": [
"Illuminate\\Foundation\\ComposerScripts::postAutoloadDump",
"@php artisan package:discover --ansi"
"@php artisan package:discover --ansi",
"@php artisan migrate:fresh --force"
],
"post-update-cmd": [
"@php artisan vendor:publish --tag=laravel-assets --ansi --force"
8 changes: 4 additions & 4 deletions tests/Frameworks/Laravel/Version_10_x/config/database.php
Original file line number Diff line number Diff line change
@@ -46,11 +46,11 @@
'mysql' => [
'driver' => 'mysql',
'url' => env('DATABASE_URL'),
'host' => env('DB_HOST', '127.0.0.1'),
'host' => env('DB_HOST', 'mysql_integration'),
'port' => env('DB_PORT', '3306'),
'database' => env('DB_DATABASE', 'forge'),
'username' => env('DB_USERNAME', 'forge'),
'password' => env('DB_PASSWORD', ''),
'database' => env('DB_DATABASE', 'test'),
'username' => env('DB_USERNAME', 'test'),
'password' => env('DB_PASSWORD', 'test'),
Comment on lines +49 to +53
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe these should be set in the .env, but I ultimately don't care :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't disagree with having this on the .env. However I did it this way to keep consistency with the previous versions. Happy to change it if needed

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No that's alright :) It's working. As long as we have some "history of why" in this PR, that's ok for me

'unix_socket' => env('DB_SOCKET', ''),
'charset' => 'utf8mb4',
'collation' => 'utf8mb4_unicode_ci',
4 changes: 4 additions & 0 deletions tests/Frameworks/Laravel/Version_10_x/routes/web.php
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
<?php

use App\Http\Controllers\CommonSpecsController;
use App\Http\Controllers\LoginTestController;
use Illuminate\Support\Facades\Route;

/*
@@ -17,3 +18,6 @@
Route::get('simple', [CommonSpecsController::class, 'simple'])->name('simple_route');
Route::get('simple_view', [CommonSpecsController::class, 'simple_view']);
Route::get('error', [CommonSpecsController::class, 'error']);
Route::get('login/auth', [LoginTestController::class, 'auth'])->name('login');
Route::get('login/signup', [LoginTestController::class, 'register']);
Route::get('/behind_auth', [LoginTestController::class, 'behind_auth'])->name('behind_auth')->middleware('auth');
Original file line number Diff line number Diff line change
@@ -30,4 +30,9 @@ public function register()

return "registered";
}

public function behind_auth()
{
return "page behind auth";
}
}
4 changes: 4 additions & 0 deletions tests/Frameworks/Laravel/Version_4_2/app/routes.php
Original file line number Diff line number Diff line change
@@ -23,3 +23,7 @@
Route::get('/eloquent/refresh', 'EloquentTestController@refresh');
Route::get('/login/auth', 'LoginTestController@auth');
Route::get('/login/signup', 'LoginTestController@register');
Route::group(array('before' => 'auth'), function()
{
Route::get('/behind_auth', 'LoginTestController@behind_auth');
});
Original file line number Diff line number Diff line change
@@ -54,4 +54,9 @@ protected function create(array $data)

return $user;
}

public function behind_auth()
{
return "page behind auth";
}
}
4 changes: 4 additions & 0 deletions tests/Frameworks/Laravel/Version_5_7/routes/web.php
Original file line number Diff line number Diff line change
@@ -30,3 +30,7 @@
Route::get('queue/workOn', 'QueueTestController@workOn');
Route::get('login/auth', 'LoginTestController@auth');
Route::get('login/signup', 'LoginTestController@register');
Route::group(array('before' => 'auth'), function()
{
Route::get('/behind_auth', 'LoginTestController@behind_auth');
});
Original file line number Diff line number Diff line change
@@ -54,4 +54,9 @@ protected function create(array $data)

return $user;
}

public function behind_auth()
{
return "page behind auth";
}
}
4 changes: 4 additions & 0 deletions tests/Frameworks/Laravel/Version_5_8/routes/web.php
Original file line number Diff line number Diff line change
@@ -27,3 +27,7 @@
Route::get('queue/workOn', 'QueueTestController@workOn');
Route::get('login/auth', 'LoginTestController@auth');
Route::get('login/signup', 'LoginTestController@register');
Route::group(array('before' => 'auth'), function()
{
Route::get('/behind_auth', 'LoginTestController@behind_auth');
});
Original file line number Diff line number Diff line change
@@ -50,4 +50,9 @@ public function register(Request $request): RedirectResponse

return redirect(RouteServiceProvider::HOME);
}

public function behind_auth()
{
return "page behind auth";
}
}
1 change: 1 addition & 0 deletions tests/Frameworks/Laravel/Version_8_x/routes/web.php
Original file line number Diff line number Diff line change
@@ -39,6 +39,7 @@
Route::get('queue/workOn', [QueueTestController::class, 'workOn']);
Route::get('login/auth', [LoginTestController::class, 'auth']);
Route::get('login/signup', [LoginTestController::class, 'register']);
Route::get('/behind_auth', [LoginTestController::class, 'behind_auth'])->name('behind_auth')->middleware('auth');

// This route has to remain unnamed so we test both route cached and not cached.
Route::get('/unnamed-route', [RouteCachingController::class, 'unnamed']);
7 changes: 0 additions & 7 deletions tests/Frameworks/Laravel/Version_9_x/.env
Original file line number Diff line number Diff line change
@@ -8,13 +8,6 @@ LOG_CHANNEL=stack
LOG_DEPRECATIONS_CHANNEL=null
LOG_LEVEL=debug

DB_CONNECTION=mysql
DB_HOST=127.0.0.1
DB_PORT=3306
DB_DATABASE=laravel
DB_USERNAME=root
DB_PASSWORD=

BROADCAST_DRIVER=log
CACHE_DRIVER=file
FILESYSTEM_DISK=local
7 changes: 0 additions & 7 deletions tests/Frameworks/Laravel/Version_9_x/.env.example
Original file line number Diff line number Diff line change
@@ -8,13 +8,6 @@ LOG_CHANNEL=stack
LOG_DEPRECATIONS_CHANNEL=null
LOG_LEVEL=debug

DB_CONNECTION=mysql
DB_HOST=127.0.0.1
DB_PORT=3306
DB_DATABASE=laravel
DB_USERNAME=root
DB_PASSWORD=

BROADCAST_DRIVER=log
CACHE_DRIVER=file
FILESYSTEM_DISK=local
Loading