-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
20 changed files
with
736 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
name: Delete old folders from GitHub Pages | ||
|
||
on: | ||
push: | ||
branches: | ||
- "gh-pages" | ||
schedule: | ||
- cron: '0 0 * * *' # This will run the workflow daily at midnight UTC | ||
|
||
jobs: | ||
delete_old_folders: | ||
runs-on: ubuntu-latest | ||
permissions: | ||
contents: write | ||
|
||
steps: | ||
- name: Checkout code | ||
uses: actions/checkout@v4 | ||
with: | ||
ref: gh-pages | ||
|
||
- name: Set up Python | ||
uses: actions/setup-python@v4 | ||
with: | ||
python-version: 3.9 | ||
|
||
- name: Get current directory | ||
run: echo "CURRENT_DIR=$(pwd)" >> $GITHUB_ENV | ||
|
||
- name: Run the script | ||
run: python rm_old_folders.py --n-days 30 --folder-name "${{ env.CURRENT_DIR }}" | ||
|
||
- name: Commit all changed files back to the repository | ||
uses: stefanzweifel/git-auto-commit-action@v5 | ||
with: | ||
branch: gh-pages | ||
commit_message: Delete folders older than 30 days | ||
|
||
notify_on_delete_pages_failure: | ||
runs-on: ubuntu-latest | ||
needs: | ||
- delete_old_folders | ||
if: failure() | ||
steps: | ||
- name: Slack Notification | ||
uses: rtCamp/action-slack-notify@v2 | ||
env: | ||
SLACK_TITLE: ":boom: The nightly delete of expired Playwright reports job has failed in ${{ github.repository }}." | ||
MSG_MINIMAL: true | ||
SLACK_WEBHOOK: ${{ secrets.SLACK_WEBHOOK }}" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,113 @@ | ||
import argparse | ||
import os | ||
import re | ||
from datetime import datetime, timedelta | ||
import shutil | ||
|
||
def find_old_folders(n_days, directory): | ||
""" | ||
Find folders in the specified directory that are older than n_days. | ||
Args: | ||
directory (str): The directory to search for folders. | ||
n_days (int): The number of days to determine which folders to delete. | ||
Returns: | ||
list: List of folder names older than n_days. | ||
""" | ||
current_time = datetime.utcnow() | ||
folder_name_regex = re.compile(r'^\d{8}_\d{6}Z$') | ||
|
||
old_folders = [] | ||
for entry in os.scandir(directory): | ||
if entry.is_dir() and re.match(folder_name_regex, entry.name): | ||
try: | ||
folder_date = datetime.strptime(entry.name, "%Y%m%d_%H%M%SZ") | ||
time_difference = current_time - folder_date | ||
if time_difference > timedelta(days=n_days): | ||
old_folders.append(entry.name) | ||
else: | ||
print( | ||
f"SKIPPED --- Folder '{entry.name}' is not older than " | ||
f"{n_days} days. It will not be deleted." | ||
) | ||
except ValueError: | ||
print( | ||
f"SKIPPED --- Error parsing timestamp for folder '{entry.name}'. " | ||
f"It will not be deleted." | ||
) | ||
else: | ||
print( | ||
f"SKIPPED --- Found folder/file with name '{entry.name}' that does " | ||
f"not match the expected timestamp format. It will not be deleted." | ||
) | ||
|
||
return old_folders | ||
|
||
def is_valid_directory(base_directory, folder_path): | ||
""" | ||
Check if the folder_path is a valid directory within the base_directory. | ||
Args: | ||
base_directory (str): The base directory. | ||
folder_path (str): The path of the folder to validate. | ||
Returns: | ||
bool: True if the folder_path is valid, False otherwise. | ||
""" | ||
# Resolve absolute paths | ||
base_directory = os.path.abspath(base_directory) | ||
folder_path = os.path.abspath(folder_path) | ||
|
||
# Ensure that the folder_path starts with the base_directory | ||
return folder_path.startswith(base_directory) | ||
|
||
def delete_folders(base_directory, folder_names): | ||
""" | ||
Delete specified folders and their contents in the given directory. | ||
Args: | ||
base_directory (str): The base directory containing the folders to delete. | ||
folder_names (list): List of folder names to delete. | ||
""" | ||
for folder_name in folder_names: | ||
folder_path = os.path.join(base_directory, folder_name) | ||
if is_valid_directory(base_directory, folder_path): | ||
try: | ||
shutil.rmtree(folder_path) | ||
print( | ||
f"DELETED --- Folder '{folder_name}' and its contents have " | ||
f"been deleted." | ||
) | ||
except FileNotFoundError: | ||
print(f"Folder '{folder_name}' not found.") | ||
except Exception as e: | ||
print(f"Error deleting folder '{folder_name}': {e}") | ||
else: | ||
print(f"SKIPPED --- Invalid folder path: '{folder_path}'") | ||
|
||
if __name__ == "__main__": | ||
parser = argparse.ArgumentParser( | ||
description="Delete old folders in a specified directory." | ||
) | ||
parser.add_argument( | ||
"--n-days", | ||
type=int, | ||
required=True, | ||
help="Number of days (days older than current date) to determine " | ||
"which folders to delete." | ||
) | ||
parser.add_argument( | ||
"--folder-name", | ||
type=str, | ||
required=True, | ||
help="Full path to the directory where reports are located." | ||
) | ||
args = parser.parse_args() | ||
|
||
# Ensure the provided folder name is an absolute path | ||
if not os.path.isabs(args.folder_name): | ||
raise ValueError("The folder name must be an absolute path.") | ||
|
||
old_folders = find_old_folders(args.n_days, args.folder_name) | ||
delete_folders(args.folder_name, old_folders) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
import { defineConfig, devices } from "@playwright/test"; | ||
import dotenv from "dotenv"; | ||
|
||
/** | ||
* Read environment variables from file. | ||
* https://github.com/motdotla/dotenv | ||
*/ | ||
dotenv.config({ path: "../.env" }); | ||
|
||
/** | ||
* See https://playwright.dev/docs/test-configuration. | ||
*/ | ||
export default defineConfig({ | ||
testDir: "playwright", | ||
testMatch: ["**/*.spec.js", "**/*.spec.ts"], | ||
/* Run tests in files in parallel */ | ||
fullyParallel: false, | ||
/* Fail the build on CI if you accidentally left test.only in the source code. */ | ||
forbidOnly: !!process.env.CI, | ||
/* Retry on CI only */ | ||
retries: process.env.CI ? 2 : 0, | ||
/* Opt out of parallel tests on CI. */ | ||
workers: process.env.CI ? 1 : undefined, | ||
/* Reporter to use. See https://playwright.dev/docs/test-reporters */ | ||
reporter: "html", | ||
/* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */ | ||
use: { | ||
/* Base URL to use in actions like `await page.goto('/')`. */ | ||
baseURL: process.env.BASE_URL || "http://localhost:3000", | ||
|
||
/* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ | ||
trace: "on-first-retry", | ||
|
||
/* Video recording configuration */ | ||
video: "retain-on-failure", | ||
}, | ||
|
||
/* Configure projects for major browsers */ | ||
projects: [ | ||
{ | ||
name: "setup", | ||
use: { ...devices["Desktop Chrome"] }, | ||
testMatch: /.*\.setup\.ts/, | ||
}, | ||
{ | ||
name: "chromium", | ||
use: { ...devices["Desktop Chrome"] }, | ||
dependencies: ["setup"], | ||
}, | ||
], | ||
|
||
/* Run your local dev server before starting the tests */ | ||
webServer: { | ||
command: process.env.CI ? "" : "cd ../ && ./run local", | ||
url: process.env.BASE_URL || "http://localhost:3000", | ||
reuseExistingServer: !!process.env.CI, | ||
stdout: "pipe", | ||
}, | ||
}); |
Oops, something went wrong.