Eu tenho um programa que, toda vez que ele faz algo, ele relê seu arquivo de configuração. Eu gostaria de substituir esse arquivo por algum tipo de arquivo virtual, que chama um script e gera os dados de configuração na hora. Eu não posso mudar a maneira como o arquivo é chamado, senão eu poderia simplesmente canalizar os dados de configuração. Então, toda vez que meu arquivo "stuff.conf" é lido, ele deve obter a saída de stuff.sh em execução.
Senão eu teria que reescrever essa configuração periodicamente, mas poderia haver horas ou segundos entre as mudanças nos dados de origem ou leituras do programa. Então eu teria que definir o timer muito baixo e provavelmente desperdiçaria recursos 90% do tempo.
Um pipe nomeado poderia ser usado para isso:
Abrir um fifo no modo somente gravação, como o shell faz
>
aqui, antes de executar,generator
trava até que outro processo o abra no modo leitura.Então
generator
será iniciado assim que o programa abrir o arquivo.Criamos um novo fifo toda vez (substituído atomicamente por
mv
) para evitar que uma segundagenerator
instância seja iniciada e alimente a saída para o mesmo pipe enquanto o programa ainda não terminou de ler a primeira saída.Isso pressupõe que o programa leia o arquivo sequencialmente e que nenhum outro processo esteja tentando lê-lo ao mesmo tempo.
De qualquer forma, você não deve fazer isso: ao sobrescrever o arquivo, você corre o risco de ele ser recarregado enquanto você o grava, e, portanto, ser lido em um estado inválido!
Em vez disso, você iria e escreveria a configuração desejada em um arquivo separado (geralmente no mesmo diretório) e então usaria
mv
(em palavras do Unix, arename
chamada do sistema) para substituir atomicamente todo o arquivo de configuração antigo por um novo .Agora,
Então, faça a substituição somente quando os dados de origem mudarem, não "sempre"? Isso me parece ser a coisa óbvia a fazer aqui.
Basta ter um trabalho em segundo plano (um "daemon") que monitora os dados de origem (como quer que isso funcione! Muitas vezes, você não precisa monitorar, pode ser notificado sobre as alterações e pode simplesmente deixar seu programa dormir até lá). E quando, e somente quando, ele muda, isso aciona a gravação de um novo arquivo de configuração (temporário) e a substituição do existente.
Agora, se isso acontece a cada dois segundos: isso é basicamente uma eternidade para um computador, então, a menos que você tenha realmente feito um benchmark e ele acabe sendo muito faminto por recursos, ... não se preocupe com isso sendo muito faminto por recursos? As escalas de tempo de humanos e computadores são tão diferentes que os humanos frequentemente se preocupam com coisas que realmente tomam um tempo tão pequeno de um computador que isso realmente não importa.
Geralmente , do ponto de vista do design do sistema, seu programa que lê o arquivo de configuração é um consumidor de dados que verifica periodicamente se há novos dados (ou seja, quando lê o arquivo de configuração).
Então, não sei qual é sua fonte e quais são seus coletores — adicionar isso à pergunta obviamente melhoraria sua capacidade de resposta — mas se for um dispositivo de entrada, ou se for algo que publica alterações por meio de barramentos de sessão dbus, ou se for algo que recebe pacotes de rede, ou mesmo se for apenas algo que grava arquivos em um diretório, essas são coisas que você pode esperar sem ter que recriar periodicamente um arquivo de configuração.
Sim, mas se seu programa for o único "usuário" do arquivo de configuração gerado, muito mais eficiente seria ter uma parte do seu "programa" existente respondendo à interface inotify(7) dentro de seu loop principal . Quando um
IN_ACCESS
evento é encontrado, ele leria todos os arquivos necessários e executaria outras operações necessárias para gerar a configuração nas estruturas de dados internas mantidas na memória do programa, então sempre que ele "fizesse algo", ele usaria as estruturas internas em vez de ler quaisquer arquivos.Alternativamente, você pode simplesmente escrever o arquivo como parte do
IN_ACCESS
manipulador de eventos inotify(7). Nesse caso, deve-se tomar cuidado para desabilitar o manipulador de eventos (ou usarIN_ONESHOT
), ou definir um sinalizador interno durante a gravação no arquivo.