Integrando uploads do WordPress com AWS S3

Utilizo estrutura da AWS para os projetos da empresa e recentemente fiz algumas otimizações em nossa infraestrutura, para projetos web utilizamos normalmente EC2 + RDS e eventualmente integrações com S3 e SES.

Para otimizar os custos com alguns projetos uma das saídas foi reduzir o storage do EC2 e passar o que era possível para o S3 – a biblioteca de mídias do WordPress é um exemplo que traz outras vantagens além da redução de custo:

  • Facilidade para migração futura
  • Facilidade de integração/migração de CDN
  • Redução da carga de requests do servidor da aplicação

Vamos para a parte técnica:

Como instalar e usar a S3 Uploads by Human Made

A biblioteca que escolhi foi a S3 Uploads, feita pela Human Made. Os motivos:

  • Gratuita e open source
  • Facilita fazer o trabalho de transferência e migração de URLs das mídias antigas
  • Funciona com qualquer tipo de mídia
  • Você configura o bucket e continua com autoridade e autonomia sobre o conteúdo e a infra

Os requisitos:

O que precisa ser feito para enviar toda a biblioteca de mídias do WordPress para o S3 usando a S3 Uploads:

  1. Configurar um IAM em sua conta da AWS, com acesso ao S3
  2. Criar uma chave de acesso para a IAM
  3. Criar um novo Bucket S3
  4. Instalar o S3 Uploads – Human Made
  5. Criar o arquivo de configuração
  6. Ativar o plugin S3 Uploads
  7. Enviar toda a biblioteca de mídia para o bucket S3

Antes de começar, faça o que é certo: backup do banco de dados do WordPress e também da instalação do seu WordPress.

Rodei essa integração com 4 sites e em apenas em um deles tive um problema – não precisei restaurar backup, mas foi necessário desativar e reinstalar o plugin.


Configurar um IAM na AWS

Acesse o Gerenciamento de IAM da AWS e clique em Adicionar.

Adicione a política AmazonS3FullAccess (você também pode criar uma restrição por Bucket para esta IAM, assim ela só terá acesso a um bucket específico) e crie o usuário.


Criar uma chave de acesso para a IAM

Na tela do usuário vá na aba Credenciais de Segurança e na área de Chaves de acesso clique em Criar chave de acesso – crie uma nova chave para aplicações rodadas fora da AWS e guarde os dados, você precisará da chave de acesso e da chave de segurança para confiruar o plugin no WordPress.


Criar um novo Bucket S3

Vá para a área de console do S3 do AWS e clique em Criar Bucket, dê um nome para seu novo bucket, mantenha ACLs habilitadas e remova o bloqueio público (o material deste bucket será público, portanto o uso dele deve ser exclusivo para as mídias públicas do seu WordPress) e concorde com a configuração do bucket público.


Instalar o S3 Uploads – Human Made

Para instalar o plugin será necessário o composer instalado e você deverá acessar (ssh) o servidor e, no diretório público do WordPress executar:

composer require humanmade/s3-uploadsCode language: JavaScript (javascript)

Isso irá instalar o s3-uploads e todas as suas dependências, o resultado esperado é que seu projeto tenha um diretório /vendor, um composer.json e dentro do diretório de plugins seja criado o diretório s3-uploads.


Criando o arquivo de configuração

A configuração dos dados do plugin é diretamente no wp-config.php pois este plugin não tem uma interface gráfica.

No seu arquivo de configuração do WordPress adicione o autoload do composer:

require_once __DIR__ . '/vendor/autoload.php';Code language: PHP (php)

Isso irá adicionar o autoload e todas as dependências que foram instaladas com o composer no carregamento da aplicação.

Agora preencha as configurações com as informações da sua conta AWS:

define( 'S3_UPLOADS_BUCKET', 'odesenvolvedor' );
define( 'S3_UPLOADS_REGION', 'us-east-1' );

define( 'S3_UPLOADS_KEY', 'iam-key' );
define( 'S3_UPLOADS_SECRET', 'iam-secret' );

define( 'S3_UPLOADS_USE_INSTANCE_PROFILE', true );Code language: JavaScript (javascript)

Ativar o plugin S3 Uploads

Para ativar o plugin você usará o WP Cli, com o comando abaixo:

wp plugin activate s3-uploads

Neste ponto os novos uploads da biblioteca de mídia do WordPress já devem estar sendo enviadas para o bucket S3 e recebendo as URLs diretamente do bucket público.

Você pode testar se a integração está funcionando com o comando:

wp s3-uploads verify

Isso irá testar inserir e excluir um objeto do bucket.


Enviar toda a biblioteca de mídia para o bucket S3

Para enviar toda a biblioteca de mídia para o S3, estando no diretório público do seu WordPress utilize o comando:

wp s3-uploads upload-directory wp-content/uploads uploads

Isso irá copiar o conteúdo de wp-content/uploads para o diretório uploads no seu bucket.

O plugin faz também o ajuste das urls de todas as mídias apontando para o S3 – mas não faz a exclusão automática das imagens antigas.

Os donos dos encanamentos

Antigamente os ricos tinham terras e o que definia o status e o poder era a quantidade de terras que se tinha. Quanto mais terra, mais vassalos. Quanto mais vassalos, mais mão de obra trabalhando e maior a sua riqueza e poder. As leis, até muito pouco tempo atrás, trabalhavam para que pouquíssimos pudessem, efetivamente, ser donos das terras.

Até muito recentemente tinha-se apenas a posse (e muita área rural ainda é assim) e na troca da posse, os donos da terra ganhavam um pouquinho – isso praticamente virou o ITBI – mas agora você é o dono e quem recolhe este imposto é o governo.

Passaram-se algumas centenas de anos e vivemos a revolução industrial – agora além das terras os ricos tinham outros meios de produzir riqueza e explorar o trabalho alheio: máquinas, fábricas, projetos industriais. Criou-se uma nova classe dos detentores do poder econômico e social: os burgueses, que se estabeleceram com as indústrias de petróleo, transportes e os mercados de capitais para iniciar o processo de empresas centenárias, perpetuando o histórico de concentração de renda.

Chegamos na era da mídia, rádio e TV agora traziam informação, entretenimento e uma nova oportunidade de exibir anúncios e vender produtos para milhares de desavisados.

Enquanto a TV se popularizava a concessão de transmissão era barata e as empresas compraram concessões e tinham lá o seu canal de TV. Eram poucos canais pois era o que era possível transmitir com a tecnologia da época – e conseguir conteúdo não era uma tarefa tão simples. Era uma transmissão direta do canal para o usuário: os canais de TV eram os donos do meio de transmissão, os encanamentos do negócio eram Canal -> usuário.

Eis que vem uma revolução no meio da mídia: a TV a cabo com uma nova maneira de levar canais para centenas de milhares de telespectadores – criou-se um novo sistema de “encanamento” para distribuição de um produto e agora os canais tinham que pagar para uma operadora para conseguirem chegar aos seus usuários – a TV a cabo passou a valer mais do que os canais de TV.

Mais alguns anos chegamos nos meios digitais, o nascimento da internet comercial e logo em seguida a efusão de esperança na democratização do conteúdo com a promessa da “web 2.0”.  O discurso agora era de que muito valor poderia ser criado mesmo se você não tivesse muito capital: uma promessa de que com uma boa ideia de produto ou serviço você poderia criar um excelente negócio.

Agora a burguesia e os donos dos “meios de produção” não tinham como impedir um programador de desenvolver e lançar um produto, ou um escritor ou jornalista de criar o seu blog e gerar e distribuir seu próprio conteúdo para um número infinito de usuários. Surgiu o RSS que fazia essa ligação direta entre público e público. O público tinha controle do que consumir e o produtor tinha finalmente um número virtualmente infinito de audiência, sem intermediários.

Os encanamentos chegaram finalmente na mão da ralé, do peão, dos trabalhadores.

E isso parecia estar funcionando: nasceram Apple, Microsoft, IBM e dezenas de empresas de jogos e software que revolucionaram o mercado computacional e de entretenimento.

Parecia, mas o que não se considerava – mas agora é óbvio em retrospecto – é a astúcia do capitalismo em cooptar inovações, gerar lucro e criar empecilhos para novos entrantes. O capitalista é um especialista em criar estes novos encanamentos, vender como se fosse algo bom para todos – ele vai chamar isso de eficiência e dizer que aumentará a disponibilização de um bem, barateará custos, ou qualquer outro discurso que fique bom em uma notícia paga. E quando o mundo aderir ao seu novo sistema de distribuição logo em breve será pego nem tão de surpresa com a instalação de um relógio medidor, que virá com uma cobrança para que qualquer um que queira utilizar aquele sistema.

Passamos pela nobreza do feudalismo, os burgueses da era industrial e me parece que carecemos de um novo termo para definir os donos dos encanamentos do mundo digital. Minha sugestão é titãs digitais, e estes trabalham incansavelmente para que tudo transite dentro do seu sistema de encanamentos: Google, Facebook, Amazon, Apple.

O Google matou o Google Reader, um leitor de Feed RSS que fazia essa ligação entre criador de conteúdo e público. Recentemente também matou o Google Podcast, que fazia esta mesma conexão com podcasts – agora você tem que usar o Youtube para isso e tanto o criador quanto o público ficam a mercê do obscuro algoritmo que define o que você vai consumir, ou para quem seu conteúdo será entregue.

Se você não utiliza os encanamentos de distribuição deles, você não está fazendo buscas e gerando impressão de anúncio, e acaba não clicando no primeiro resultado patrocinado – da loja que você já conhece mas agora reparte uma considerável do lucro com o Google via adwords.

E os donos dos encanamentos farão de tudo para que nenhum novo sistema de encanamento apareça – e vão gastar recursos com lobby, tecnologia e manipulação de opinião para fechar qualquer gotinha que esteja vazando para fora.

Como testar localmente uma Google Cloud Function com trigger de tópico PubSub

Estou trabalhando em um projeto de monitoramento de dados com Google Cloud Functions e, para agilizar e evitar custos desnecessários, queria testar a ativação de uma GCF com um gatilho de evento (tópico no PubSub)

O primeiro passo é iniciar o emulador de PubSub, nesse caso ele vai rodar na porta 8043, para o projeto odesenvolvedor

Criando um emulador do PubSub

gcloud beta emulators pubsub start \
    --project=odesenvolvedor \
    --host-port='localhost:8043'Code language: JavaScript (javascript)

É esperado que você veja no terminal o log do emulador, com alguns detalhes da exeução e indicando que ele está rodando e em qual porta:

[pubsub] INFO: Server started, listening on 8043Code language: CSS (css)

Você vai precisar criar um tópico, em outro terminal execute:

curl -s -X PUT 'http://localhost:8043/v1/projects/odesenvolvedor/topics/mytopic'Code language: JavaScript (javascript)

Agora vamos especificar o endpoint de assinatura do push

curl -s -X PUT 'http://localhost:8043/v1/projects/odesenvolvedor/subscriptions/mysub' \
    -H 'Content-Type: application/json' \
    --data '{"topic":"projects/odesenvolvedor/topics/mytopic","pushConfig":{"pushEndpoint":"http://localhost:8080/projects/odesenvolvedor/topics/mytopic"}}'Code language: JavaScript (javascript)

Enviando uma mensagem para o PubSub

Com o emulador rodando você pode disparar novas mensagens fazendo um post para o emulador (no nosso exmeplo, localhost:8043) indicando o projeto, tópico e ação.

Para publicar uma mensagem no terminal, você pode usar o comando abaixo, lembrando que o data da mensagem deve estar codificado em base64.

curl -s -X POST 'http://localhost:8043/v1/projects/odesenvolvedor/topics/mytopic:publish' \
    -H 'Content-Type: application/json' \
    --data '{"messages":[{"data":"eyJmb28iOiJiYXIifQ=="}]}'Code language: JavaScript (javascript)

Você também pode usar uma ferramenta como Postman para fazer os envios, configurando método como POST, a url como http://localhost:8043/v1/projects/odesenvolvedor/topics/mytopic:publish e o tipo de conteúdo como application/json, enviando a mensagem no corpo do post.

Rodando uma Cloud Function local que escute o evento

Antes de criar a função, você precisa definir as variáveis de ambiente para que sua função utilize o PubSub do emulador e não tente se conectar ao PubSub da Google Cloud, para isso execute o comando abaixo, trocando o id do projeto:

gcloud beta emulators pubsub env-init
export PUBSUB_EMULATOR_HOST=[::1]:8432
export PUBSUB_PROJECT_ID=odesenvolvedorCode language: JavaScript (javascript)

Com o emulador rodando e sabendo como enviar uma mensagem para um tópico, falta apenas criarmos uma função que ouça o tópico.

Crie a função na sua linguagem favorita, no exemplo abaixo uma função em Python:

import json
import base64

def minha_gcf(event, _context = None):
    payload = base64.b64decode(event['data']).decode('utf-8')
    payload = json.loads(payload)
    print(payload)
    return "OK"Code language: JavaScript (javascript)

Agora é só rodar esta função localmente, avisando que ela deve ouvir os eventos do PubSub:

functions_framework --debug --port 8080 --target=minha_gcf --signature-type=event

O importante aí é o –signature-type=event, mesmo sendo uma função local rodando na porta 8080 ela irá escutar os eventos do PubSub

Ao enviar novas mensagens para seu emulador do PubSub, sua função será executada.

ISPConfig 3.2 e jailkit – problemas para login com ssh

Depois de atualizar o ISPConfig para a versão 3.2 tive problemas com alguns acessos ssh para usuários com jailkit ativo.

Aqui, alguns usuários logo após tentar fazer o login com chave, eram desconectados do ssh.

Como a maioria dos processos de deploy que utilizo neste servidor utilizam rsync, isso travou o deploy das aplicações com este problema.

Baseado neste link e em algumas outras leituras, acompanhei o /var/log/auth.log enquanto autenticava com os usuários e parecia ser um erro de permissão, o resultado era muito parecido com o da thread linkada:

Oct 15 18:46:43 server1 jk_chrootsh[16809]: ERROR: failed to execute shell /bin/bash for user username (5007), check the permissions and libraries of /var/www/clients/client1/web78//bin/bashCode language: JavaScript (javascript)

Logo após isso, o logout do usuário. A barra dupla não é um problema (foi a primeira coisa que achei que poderia ser).

No fim, depois de muitas horas e testes, eu não consegui identificar a causa do problema. As configurações do jailkit estava idênticas as ispconfig.

Tentei fazer o resync dos usuários shell, não retornava erro mas não alterava em nada.

Solução encontrada

O que resolveu aqui, foi bem parecido com o sugerido no final da thread no fórum:

  • Removi todos os usuários shell da conta em questão
  • Na configuração de site, marque na aba options a opção Delete unused jailkit chroot
  • Aguarde a sincronização do ISPConfig ou execute manualmente o update via .sh no server
  • Recrie o usuário shell – isso irá recriar a estrutura de arquivos para o chroot e recriar o usuário

Estes passos resolveram e o acesso voltou a funcionar com o jailkit, mas não consegui identificar a causa (não ocorreu com todos os usuários com jailkit).

Mantendo o WordPress seguro – permissões corretas de arquivos e diretórios

Seguindo as recomendações de segurança do WordPress, as permissões de arquivos devem ser: 755 para diretórios e 644 para arquivos.

Isso pode ser configurado com o comando abaixo (unix):

find . -type d -exec chmod 755 {} \;
find . -type f -exec chmod 644 {} \;Code language: CSS (css)

Para os uploads funcionarem (via wp-admin) é necessário que o usuário do seu servidor (no meu caso, o Apache usa o www-data) tenha acesso a pasta de uploads.

chown www-data:www-data wp-content/uploads

O acesso ao www-data também é necessário no restante dos arquivos se você quiser habilitar o update via painel, instalação de temas e plugins.