Welcome to the Vanguard Development Guide! We're happy that you're interested in contributing to Vanguard.
This guide will help you:
- Set up your development environment
- Understand our contribution process
- Start contributing to Vanguard effectively
We aim to make your onboarding process smooth and productive. Let's get started!
Vanguard is an open-source backup solution for servers and applications. Built with Laravel, it's designed by the community for the community to simplify data protection.
In the following sections, we'll guide you through setting up your development environment, understanding our workflow, and making your first contribution. Let's get started!
This section will guide you through setting up your development environment for Vanguard.
Before you begin, ensure you have the following installed on your system:
- PHP 8.2 or higher
- Composer
- Node.js (v14 or higher) and npm
- Git
While you can use any development environment you're comfortable with, we recommend the following options for a smooth development experience:
Laravel Valet is a lightweight development environment for macOS. It's easy to set up and uses minimal system resources.
To install:
brew install php
composer global require laravel/valet
valet install
Laravel Herd is a native macOS application that provides a complete PHP development environment. It's user-friendly and requires no configuration.
To install, download it from the official Laravel Herd website.
Laragon is an isolated, portable development environment for Windows. It's easy to use and comes with a variety of tools pre-installed.
::: tip Regardless of your chosen environment, ensure it meets Vanguard's PHP version and extension requirements. :::
- Fork the Vanguard repository on GitHub.
- Clone your fork locally:
git clone https://github.com/vanguardbackup/vanguard.git cd vanguard
- Install PHP dependencies:
composer install
- Install JavaScript dependencies:
npm install
- Set up your environment:
cp .env.example .env php artisan key:generate
- Configure your
.env
file with your local database details.
Set up your database and seed it with test data:
php artisan migrate:fresh --seed
- Start the development server (skip if using Valet/Herd):
php artisan serve
- In a separate terminal, compile assets:
npm run dev
You should now be able to access the Vanguard application at http://localhost:8000
.
Congratulations! You've successfully set up your Vanguard development environment. In the next section, we'll cover the development workflow and how to make your first contribution.
- Create a new branch for your feature or bug fix:
git checkout -b feature/your-feature-name
- Make your changes, following our coding standards (PSR-12).
- Run tests and ensure they pass:
php artisan test
- Commit your changes with a clear, descriptive commit message.
- Push your branch to your fork:
git push origin feature/your-feature-name
- Submit a Pull Request (PR) to the main Vanguard repository.
Vanguard employs several tools to maintain high code quality:
-
Duster: Automated code formatter combining TLint, PHP_CodeSniffer, PhpCsFixer, and Laravel Pint to enforce consistent coding style.
-
PHPStan: Static analysis tool to detect potential errors and improve code quality.
-
Rector: Performs instant upgrades and automated refactoring of PHP code.
-
Husky: Executes pre-commit hooks for linting and testing, ensuring only quality code is committed.
-
GitHub Actions: Automates CI/CD processes, running tests and checks on PR submissions.
Husky runs these checks before each commit:
- PHPStan (Static analysis)
- Pest (PHP testing framework)
- Dusk (Browser testing)
- Rector (Code refactoring)
- Duster (Code formatting)
- Front-end asset build
Developers can run these checks locally:
# Static analysis
./vendor/bin/phpstan analyse
# Pest test suite
./vendor/bin/pest --parallel
# Dusk test suite
php artisan dusk --dirty
# Check for potential refactorings
./vendor/bin/rector process --dry-run
# Fix coding style issues
./vendor/bin/duster fix
# Npm asset building
npm install && npm run build
:::tip Running these checks locally helps catch and fix issues before they're flagged by the pre-commit script or CI system, streamlining the development process. :::
Thorough testing is a critical part of the development process in Vanguard. We use multiple testing frameworks to ensure comprehensive coverage:
Vanguard primarily uses Pest for unit and feature testing.
- Write tests for all new features and bug fixes.
- Ensure all tests pass before submitting a PR:
php artisan test
For browser testing, we use Laravel Dusk. This allows us to test JavaScript-dependent functionality and user interactions.
- Write Dusk tests for features that involve complex front-end interactions.
- Run Dusk tests separately:
php artisan dusk
Testing is integral to Vanguard's development process:
- Quality Assurance: Catches bugs early and prevents regressions.
- Living Documentation: Tests demonstrate expected behavior of project components.
- Refactoring Confidence: Enables easier refactoring and optimization.
- CI/CD: Our CI pipeline runs all tests before merging.
::: tip Aim for high test coverage, but focus on testing critical paths and edge cases rather than aiming for an arbitrary coverage percentage. :::
::: warning Never submit a PR without adding appropriate tests and ensuring all existing tests pass. :::
Remember, a well-tested codebase is a stable and maintainable codebase. Your tests are as important as your feature code!
Vanguard supports multiple languages. Follow these guidelines for language strings:
-
Use English strings directly in your code:
{{ __('Hello world!') }}
or
@lang('Hello world!')
-
For languages other than English, add translations to
lang/[lang_code].json
files. For example, inlang/es.json
:{ "Hello world!": "¡Hola mundo!" }
-
When adding a new language, create a new JSON file (e.g.,
fr.json
for French) and update theavailable_languages
array inconfig/app.php
.
::: warning
Never hardcode non-English strings directly in your code. Always use the __()
or @lang()
functions with English strings, and provide translations in the JSON files.
:::
::: tip Remember to keep your English strings consistent across the codebase. This makes it easier to manage translations and ensures a coherent user experience. :::
We use structured commit messages for a clear, traceable history. Each commit should:
- Start with a type prefix (e.g.,
fix:
,feat:
,refactor:
) - Be no longer than 50 characters
- End with a period
- Clearly describe the change
Examples of valid commit messages:
feat: Add user authentication feature.
fix: Resolve database connection timeout issue.
refactor: Optimize backup process for better performance.
::: info Husky is configured to run a pre-commit hook that checks your commit messages against these guidelines. It ensures that each commit message:
- Begins with an approved type prefix
- Is under 50 characters
- Ends with a period
- Follows our formatting rules
The script attempts to auto-fix non-compliant commits, but manual edits may be needed. :::
This approach aids in changelog generation, change tracking, and maintaining a clean history.
This script runs automatically before each local commit. To ensure it functions properly, verify that Husky hooks are activated in your development environment.
Maintaining up-to-date and comprehensive documentation is important. This section outlines our documentation practices and guidelines.
-
When to Update: Update documentation whenever you add, modify, or remove features.
-
Where to Update:
- In-code documentation: Update docblocks and comments in the codebase.
- The documentation site (this website!) at the GitHub repository here.
-
Documentation PRs: Submit Pull Requests to this repository when necessary. The repositories are separate.
- Clear and Concise: Write in simple, easy-to-understand language.
- Keep it Current: Regularly review and update existing documentation.
- Accessibility: Ensure documentation is accessible, including proper heading structure and alt text for images.
Adhering to best practices ensures that Vanguard remains maintainable, efficient, and secure. Here are key best practices to follow:
- SOLID Principles: Follow SOLID principles in your object-oriented design.
- DRY (Don't Repeat Yourself): Avoid code duplication; extract reusable code into functions or classes.
- Single Responsibility: Each class or function should have a single, well-defined purpose.
- Database Optimization:
- Use eager loading to prevent N+1 query problems.
- Index frequently queried columns.
- Caching: Implement caching for expensive operations or frequently accessed data.
- Asynchronous Processing: Use Laravel's queue system for time-consuming tasks.
- Input Validation: Always validate and sanitize user input.
- Authentication and Authorization: Use Laravel's built-in authentication system and Gates/Policies for authorization.
- Avoid Raw SQL: Use Eloquent throughout the project unless you need to perform a complex query that cannot be done with Eloquent in a clean manner.
- Encrypt Sensitive Data: Always encrypt sensitive information like passwords and API keys before storing in the database. Use Laravel's built-in encryption features.
- Remote Commands: When writing commands that will run on remote servers with phpseclib, be extremely cautious. Validate and sanitize all inputs, use the principle of the least privilege, and avoid passing sensitive data as command-line arguments.
- Keep PRs Focused: Each PR should address a single feature or bug fix.
- Write Self-Documenting Code: Use meaningful variable and function names.
- Commit Often: Make small, frequent commits with clear messages.
By following these best practices, you'll contribute to keeping Vanguard's codebase clean, efficient, and secure.
Here is an example of how you would contact a remote server and run a command on it. The GitHub Repository is always the best place for up-to-date, real examples of the code used in Vanguard.
<?php
namespace Example;
use App\Facades\ServerConnection;
use App\Models\RemoteServer;
use App\Support\ServerConnection\Exceptions\ConnectionException;
use Log;
class ExampleClass {
/**
* @param RemoteServer $remoteServer The remote server model.
* @return void
* @throws ConnectionException
*/
public function interactWithServer(RemoteServer $remoteServer): void
{
$serverConnection = ServerConnection::connectFromModel($remoteServer)
->establish();
$commandOutput = $serverConnection->run('whoami');
// Send the command output to Laravel's native logger etc.
Log::debug($commandOutput);
}
}
Vanguard uses feature flags to manage the rollout of new features and conduct experiments. We utilize Laravel Pennant for implementing feature flags in our application.
Feature flags (also known as feature toggles) allow us to enable or disable features in our application without deploying new code. This approach provides several benefits:
- Gradual rollout of new features
- Easy rollback if issues are discovered
- Ability to test features in production with a limited audience
Laravel Pennant provides a simple, expressive API for working with feature flags. Here's a basic example of how we use feature flags in Vanguard:
// Using a feature flag
if (Feature::active('new-dashboard')) {
return view('new-dashboard');
}
return view('old-dashboard');
For more detailed information on using Laravel Pennant, refer to the official documentation.
When deciding whether to implement a feature flag, consider the following criteria:
- Impact: Will the feature significantly change user experience or system behavior?
- Risk: Is there a potential for the feature to cause issues in production?
- Scope: Does the feature affect a large portion of the codebase or user base?
- Testing: Would it be beneficial to test the feature with a subset of users before full rollout?
If a feature meets one or more of these criteria, it's a good candidate for a feature flag.
Vanguard includes an Experiments Manager, accessible to all users, where feature flags can be toggled on or off. This interface allows for easy management of feature flags without requiring code changes.
In Vanguard, we define features using an array that includes additional metadata. This approach allows us to pass more data to the Experiments Manager, providing a richer interface for managing features. Here's how we define features:
private function defineFeatures(): void
{
$features = [
'navigation-redesign' => [
'title' => 'Navigation Redesign',
'description' => 'A new, more intuitive navigation structure for improved user experience.',
'group' => 'UI/UX',
'icon' => 'heroicon-o-bars-3',
],
'advanced-search' => [
'title' => 'Advanced Search',
'description' => 'Enhanced search functionality with filters and sorting options.',
'group' => 'Search',
'icon' => 'heroicon-o-magnifying-glass',
],
// Add more features as needed
];
}
The Experiments Manager can then use this data to create a user-friendly interface for toggling features and displaying relevant information about each feature.
- Clean Code: Avoid deeply nested conditionals based on feature flags. Instead, use separate methods or classes for different feature versions.
- Temporary Nature: Remember that feature flags should be temporary. Once a feature is fully rolled out and stable, remove the associated flag.
- Testing: Write tests for both the enabled and disabled states of a feature flag.
- Documentation: Keep an up-to-date list of active feature flags and their purposes.
- Metadata Management: When adding new features, ensure that all necessary metadata fields are included to provide a comprehensive view in the Experiments Manager.
By following these guidelines and utilizing the rich metadata approach, we can effectively use feature flags to improve our development process and user experience in Vanguard, while maintaining an informative Experiments Manager.