GitHub Actions e a Magia dos Triggers: Automatizando Tarefas com C#

GitHub Actions e a Magia dos Triggers: Automatizando Tarefas com C#

A Ideia

Um dia desses eu estava criando um processo de deploy com Github Actions e me lembrei de um sistema de ITSM que trabalhei que permitia a execução de arquivos de scripts baseado em triggers do sistema.

E seguindo essa lógica de alteração em sistemas de arquivo que o Github Actions tem, comecei a pensar em algumas possibilidades:

Automatizar backups em horários específicos.
Realizar upload automáticos de arquivos em determinada pasta
Limpeza Automática de Disco
Etc.

Então, abri meu Visual Studio e comecei a criação desse um mini projeto: O JobExecutor (criativo, né?)

A ideia, na verdade é bem simples: teria um arquivo onde nós armazenariamos as triggers ligadas ao script a ser executado.

Por enquanto, como é apenas uma prova de conceito, decidi usar somente dois tipos de trigger:

CronExpression: Que é uma forma de você informar periodicidades e é definido por uma string que tem esse formato: * * * * *

FileWatcher: Para que seja executado assim que algum arquivo ou pasta sejam alterados.

Para armazenar esses dados, escolhi o JSON e ficou nesse formato aqui:

{
“triggers”: [
{

“type”: “FileWatcher”,
“scriptFileName”: “PATH\TO\FILE.ps1″,
“watchedPath”: “PATH\TO\WATCHED\FILES”
},
{
“type”: “CronExpression”,
“scriptFileName”: “PATH\TO\FILE.ps1″,
“CronExpression”: “0 2 * * *” // Vai rodar todos os dias às 2hrs da manhã
}
]
}

Se você é atento, percebeu que nos dois scriptFileName eu estou colocando arquivos ps1 que são arquivos de código Poweshell.
Isso é porque pretendo enviar parâmetros para que o usuário possa ter mais detalhes sobre a ação e tomar decisões sobre elas e os scrips de Powershell são completos e simples de se usar.

As estruturas de código

Para esse projeto estou usando C#, e como ele é fortemente tipado, encontrar estruturas diferentes dentro do mesmo array seria um problema para ele.
E como não podemos também esperar que o usuário coloque todos os parâmetros que ele não vai precisar, não podemos simplesmente converter de JSON pra objeto diretamente.

Então, criei quatro classes:

A primeira tem a inteção de ser a classe “pai”, contendo o que todas terão:

public class Trigger
{
public string ScriptFileName { get; set; }
public string Type { get; set; }
}

A CronJobTrigger é uma implementação da Trigger.cs e vai adicionar somente o CronExpression:

public class CronJobTrigger : Trigger
{
public string CronExpression { get; set; }
}

A FileWatcherJobTrigger será também uma implementação da Trigger.cs, e vai ter o caminho que vai ser vigiado.

public class FileWatcherJobTrigger : Trigger
{
public string WatchedPath { get; set; }
}

E por fim, a que vai representar o JSON como um todo:

public class TriggerConfig
{
public Trigger[] Triggers { get; set; }
}

Convertendo os diferente tipos de Trigger

Como o FileWatcher e o CronExpression tem parâmetros diferentes, não posso simplemente convertê-los diretamente, já que o conversor vai usar só o tipo Trigger e ignorar os outros campos.

Então precisei criar um conversor.

Estamos usando o conversor do Newtonsoft

// TriggerConverter.cs
using Newtonsoft.Json.Converters;
using Newtonsoft.Json.Linq;
using Newtonsoft.Json;

namespace JobExecutor.Structs;

public class TriggerConverter : CustomCreationConverter<Trigger>
{
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
var jsonObject = JObject.Load(reader);
Trigger trigger;

if (jsonObject[“CronExpression”] != null)
{
trigger = new CronJobTrigger();
}
else if (jsonObject[“WatchedPath”] != null)
{
trigger = new FileWatcherJobTrigger();
}
else
{
throw new JsonSerializationException(“Unknown trigger type”);
}

serializer.Populate(jsonObject.CreateReader(), trigger);
return trigger;
}
}

Aqui, quando o conversor for passar de item por item ele vai verificar se o campo CronExpression está preenchido, e caso sim, vai definir o registro como do tipo CronJobTrigger.

O mesmo se aplica para o FileWatcherJobTrigger e o campo WatchedPath.

Caso nenhum dos dois seja verdade, dispara uma Exception.

O Programa

Pra começar, precisamos capturar o arquivo de triggers e ler o conteúdo dele:

class Program
{
private static List<Trigger> triggers;
private static string configPath = “PATH\TO\triggers.json”;

static void Main(string[] args)
{
LoadTriggers();
Console.ReadLine();
}

private static void LoadTriggers()
{
triggers = ReadTriggerConfig().Triggers.ToList();
}

private static TriggerConfig ReadTriggerConfig()
{

var settings = new JsonSerializerSettings
{
// Adicionamos o nosso conversor aqui
Converters = new List<JsonConverter> { new TriggerConverter() }
};

// Aqui, lemos o arquivo com um stream
using var stream = new StreamReader(configPath);
var json = stream.ReadToEnd();

return JsonConvert.DeserializeObject<TriggerConfig>(json, settings);
}
}

Com as triggers armazenadas na propriedade triggers, podemos criar o método que vai processar as triggers e executá-las.

private static void InitializeTriggers()
{
foreach (var trigger in triggers)
{
switch (trigger.Type)
{
case “CronExpression”:
SetupCronJob(trigger as CronJobTrigger);
break;
case “FileWatcher”:
SetupFileWatcher(trigger as FileWatcherJobTrigger);
break;
default:
throw new NotImplementedException($”Trigger type {trigger.Type} is not implemented.”);
}
}
}

Depois disso, vamos implementar o SetupCronJob.

Criando o Setup do CronExpression

A expressão Cron é umna string que funciona assim:

Com 5 caracteres:

* * * * *
– – – – –
| | | | |
| | | | +—– Dia da Semana(0 – 6)
| | | +——- Mês (1 – 12)
| | +——— Dia do Mês (1 – 31)
| +———– Hora (0 – 23)
+————- Minuto (0 – 59)

Com 6 Catacteres

* * * * * *
– – – – – –
| | | | | |
| | | | | +— Dia da Semana (0 – 6)
| | | | +—– Mês (1 – 12)
| | | +——- Dia do Mês (1 – 31)
| | +——— Hora (0 – 23)
| +———– Minuto (0 – 59)
+————- Segundo (0 – 59)

Com ele você pode dizer “qualquer valor” com um *, usar uma , pra definir vários valores ou até um range de valores com um -.
Exemplos:

0 12 * * *” # Todos os dias às 12:00
5 0 * 8 *” # Às 00:05, todos os dias de Agosto
15 14 1 * *” # Todo dia 1º às 14:15
*/5 * * * * *” # À cada 5 segundos

Sabendo disso, precisamos converter isso em um agendamento no nosso código.
Pra isso, vamos usar a lib NCrontab,

Vamos usar o método Parse do CrontabSchedule e em seguida armazenar a próxima execução.
Vai ficar assim:

var schedule = CrontabSchedule.Parse(trigger.CronExpression, new CrontabSchedule.ParseOptions() { IncludingSeconds = true });
var nextRun = schedule.GetNextOccurrence(DateTime.Now);

E depois, vamos criar um System.Threading.Timer para agendar a execução do método que irá rodar o script e então reagendar o job novamente.

O método completo fica assim:

private static void SetupCronJob(CronJobTrigger trigger)
{
var schedule = CrontabSchedule.Parse(trigger.CronExpression, new CrontabSchedule.ParseOptions() { IncludingSeconds = true });
var nextRun = schedule.GetNextOccurrence(DateTime.Now);
var timer = new Timer(
(e) => {
// Executa o arquivo
ExecuteScript(trigger);

// Reagenda o job
SetupCronJob(trigger);
},
null,
(long)(nextRun DateTime.Now).TotalMilliseconds,
Timeout.Infinite);
}

Criando o Setup do FileWatcher

Para verificar as alterações no caminho indicado, nós vamos usar o FileSystemWatcher.

var watcher = new FileSystemWatcher
{
Path = trigger.WatchedPath, // Caminho do arquivo pego na trigger
Filter = “*”, // Monitoraremos todos os arquivos
IncludeSubdirectories = true, // Incluiremos o monitoramento de subdiretórios
EnableRaisingEvents = true, // Permitiremos a execução de eventos
};

// Define os eventos que devem ser notificados
watcher.NotifyFilter = NotifyFilters.LastWrite
| NotifyFilters.FileName
| NotifyFilters.DirectoryName
| NotifyFilters.Attributes;

Agora, vamos criar o método que vai executar o script e adicioná-lo aos métodos watcher.Changed, watcher.Created, watcher.Deleted e watcher.Renamed do Watcher.

O Evento de edição pode (e vai) disparar mais de um evento Changed então, vamos criar um cache para isso.

private static Dictionary<string, DateTime> scriptExecutionCache = new Dictionary<string, DateTime>();

private static void SetupFileWatcher(FileWatcherJobTrigger trigger)
{
var watcher = new FileSystemWatcher
{
Path = trigger.WatchedPath,
Filter = “*”,
IncludeSubdirectories = true, // Incluiremos o monitoramento de subdiretórios
EnableRaisingEvents = true, // Permitiremos a execução de eventos
};

watcher.NotifyFilter = NotifyFilters.LastWrite
| NotifyFilters.FileName
| NotifyFilters.DirectoryName
| NotifyFilters.Attributes;

void OnChange(object sender, FileSystemEventArgs e)
{
// Verifica se pode executar
if (ScriptCanBeRunned(e.FullPath))
{
return;
}

// Executa o Script
ExecuteScript(trigger);

// Registra a ultima execução
CacheScriptExecution(e.FullPath);
}

watcher.Changed += OnChange;
watcher.Created += OnChange;
watcher.Deleted += OnChange;
watcher.Renamed += OnChange;
}

private static void CacheScriptExecution(string path)
{
scriptExecutionCache[path] = DateTime.Now;
}

private static bool ScriptCanBeRunned(string path)
{
if (!scriptExecutionCache.ContainsKey(path))
{
return false;
}

var lastExecution = scriptExecutionCache[path];

// Caso tenha executado há mais de 1 segundo, retorna `true`
return lastExecution.AddSeconds(1) > DateTime.Now;
}

O método ExecuteScript

Ele, na verdade, é bem simples.

Só vai verificar a existência e extensão do arquivo e executálo usando o Process.

private static void ExecuteScript(Trigger trigger)
{
if (!File.Exists(trigger.ScriptFileName))
{
Console.WriteLine($”File not found: {trigger.ScriptFileName});
return;
}

if (!trigger.ScriptFileName.EndsWith(“.ps1”))
{
throw new NotImplementedException($”Script type {trigger.ScriptFileName} is not implemented.”);

}

var startInfo = new ProcessStartInfo()
{
FileName = “powershell.exe”,
Arguments = $”-ExecutionPolicy Bypass -File “{trigger.ScriptFileName}“”
};

Process.Start(startInfo);
}

Depois disso, você só precisará criar o seu script e criar uma trigger para ele.

Caso não saiba fazer scripts com Powershell, leia esse artigo: about_Scripts.

Nesse exemplo, vou criar uma trigger que vai rodar à cada 5 segundos e executar um script que mostra o horário atual.

A trigger

{
“Triggers”: [
{
“ScriptFileName”: “PATH\TO\script.ps1″,
“Type”: “CronExpression”,
“CronExpression”: “*/5 * * * * *”
}
]
}

O script:

Add-Type -AssemblyName System.Windows.Forms
$now = Get-Date

Resultado:

Melhorias

Uma coisa interessante que pode ser feita é: passar parâmetros para o script.

Ex: Quando um arquivo for adicionado em uma pasta em específica, enviamos para o script o nome do evento e o nome do arquivo.

Passando argumentos

Pra isso, vamos alterar o ExecuteScript pra receber esses parâmetros:

// Recebe os parâmetros como uma lista de strings
private static void ExecuteScript(Trigger trigger, params string[] parameters)
{
if (!File.Exists(trigger.ScriptFileName))
{
Console.WriteLine($”File not found: {trigger.ScriptFileName});
return;
}

if (!trigger.ScriptFileName.EndsWith(“.ps1”))
{
throw new NotImplementedException($”Script type {trigger.ScriptFileName} is not implemented.”);
}

var startInfo = new ProcessStartInfo()
{
FileName = “powershell.exe”,
// Adiciona os parametros no final dos argumentos
Arguments = $”-ExecutionPolicy Bypass -File “{trigger.ScriptFileName}{string.Join(‘ ‘, parameters)},
};

Process.Start(startInfo);
}

OnChange:
Nele, nós passamos os parâmetros nomeados nesse formado : -{KEY} “{VALUE}” cmo abaixo

void OnChange(object sender, FileSystemEventArgs e)
{
Console.WriteLine($”File {e.Name} {e.ChangeType});
if (ScriptCanBeRunned(e.FullPath))
{
return;
}

ExecuteScript(
trigger,
$”-EventType “{e.ChangeType}“”,
$”-Name “{e.Name}“”,
$”-FullPath “{e.FullPath}“”);

CacheScriptExecution(e.FullPath);
}

E no script, nós receberemos assim:

param (
[string]$EventType,
[string]$Name,
[string]$FullPath
)

Add-Type -AssemblyName System.Windows.Forms

[System.Windows.Forms.MessageBox]::Show(“EventType: $EventType`nName: $Name`nFullPath: $FullPath”)

Resultado:

Conclusão

Ainda há muito que pode ser melhorado nesse projeto como:

[x] Criar o arquivo triggers.json caso ele não exista.
[x] Adicionar um FileSystemWatcher no arquivo triggers.json para atualizar as triggers.
[ ] Capturar o estado da máquina para poder passar em argumentos para os scripts (Ex: “MemoryUsage” ou “BatteryCharge”)
[ ] Implementar Argumentos para os Jobs de CronExpression
[ ] Implementar novos tipos de trigger como baseadas em Eventos do Windows, Emails e etc.
[ ] Transformar em serviço
[ ] Adicionar sistema de logs
[ ] Criar interface para adicionar triggers e scripts

E se você se sentir à vontade, pode me ajudar com a implementação.
Ele já está lá no Github:

Link do repositório

Leave a Reply

Your email address will not be published. Required fields are marked *