- #!/usr/bin/perl
- package IkiWiki::Plugin::amazon_s3;
- use warnings;
- no warnings 'redefine';
- use strict;
- use IkiWiki 2.00;
- use IkiWiki::Render;
- use Net::Amazon::S3;
- # Store references to real subs before overriding them.
- our %subs;
- BEGIN {
- foreach my $sub (qw{IkiWiki::writefile IkiWiki::prune}) {
- $subs{$sub}=\&$sub;
- }
- };
- sub import {
- hook(type => "getopt", id => "amazon_s3", call => \&getopt);
- hook(type => "getsetup", id => "amazon_s3", call => \&getsetup);
- hook(type => "checkconfig", id => "amazon_s3", call => \&checkconfig);
- }
- sub getopt () {
- eval q{use Getopt::Long};
- error($@) if $@;
- Getopt::Long::Configure('pass_through');
- GetOptions("delete-bucket" => sub {
- my $bucket=getbucket();
- debug(gettext("deleting bucket.."));
- my $resp = $bucket->list_all or die $bucket->err . ": " . $bucket->errstr;
- foreach my $key (@{$resp->{keys}}) {
- debug("\t".$key->{key});
- $bucket->delete_key($key->{key}) or die $bucket->err . ": " . $bucket->errstr;
- }
- $bucket->delete_bucket or die $bucket->err . ": " . $bucket->errstr;
- debug(gettext("done"));
- exit(0);
- });
- }
- sub getsetup () {
- return
- plugin => {
- safe => 0,
- rebuild => 0,
- },
- amazon_s3_key_id => {
- type => "string",
- example => "XXXXXXXXXXXXXXXXXXXX",
- description => "public access key id",
- safe => 1,
- rebuild => 0,
- },
- amazon_s3_key_id => {
- type => "string",
- example => "$ENV{HOME}/.s3_key",
- description => "file holding secret key (must not be readable by others!)",
- safe => 0, # ikiwiki reads this file
- rebuild => 0,
- },
- amazon_s3_bucket => {
- type => "string",
- example => "mywiki",
- description => "globally unique name of bucket to store wiki in",
- safe => 1,
- rebuild => 1,
- },
- amazon_s3_prefix => {
- type => "string",
- example => "wiki/",
- description => "a prefix to prepend to each page name",
- safe => 1,
- rebuild => 1,
- },
- amazon_s3_location => {
- type => "string",
- example => "EU",
- description => "which S3 datacenter to use (leave blank for default)",
- safe => 1,
- rebuild => 1,
- },
- amazon_s3_dupindex => {
- type => "boolean",
- example => 0,
- description => "store each index file twice? (allows urls ending in \"/index.html\" and \"/\")",
- safe => 1,
- rebuild => 1,
- },
- }
- sub checkconfig {
- foreach my $field (qw{amazon_s3_key_id amazon_s3_key_file
- amazon_s3_bucket}) {
- if (! exists $config{$field} || ! defined $config{$field}) {
- error(sprintf(gettext("Must specify %s"), $field));
- }
- }
- if (! exists $config{amazon_s3_prefix} ||
- ! defined $config{amazon_s3_prefix}) {
- $config{amazon_s3_prefix}="wiki/";
- }
- }
- {
- my $bucket;
- sub getbucket {
- return $bucket if defined $bucket;
-
- open(IN, "<", $config{amazon_s3_key_file}) || error($config{amazon_s3_key_file}.": ".$!);
- my $key=<IN>;
- chomp $key;
- close IN;
- my $s3=Net::Amazon::S3->new({
- aws_access_key_id => $config{amazon_s3_key_id},
- aws_secret_access_key => $key,
- retry => 1,
- });
- # make sure the bucket exists
- if (exists $config{amazon_s3_location}) {
- $bucket=$s3->add_bucket({
- bucket => $config{amazon_s3_bucket},
- location_constraint => $config{amazon_s3_location},
- });
- }
- else {
- $bucket=$s3->add_bucket({
- bucket => $config{amazon_s3_bucket},
- });
- }
- if (! $bucket) {
- error(gettext("Failed to create bucket in S3: ").
- $s3->err.": ".$s3->errstr."\n");
- }
- return $bucket;
- }
- }
- # Given a file, return any S3 keys associated with it.
- sub file2keys ($) {
- my $file=shift;
- my @keys;
- if ($file =~ /^\Q$config{destdir}\/\E(.*)/) {
- push @keys, $config{amazon_s3_prefix}.$1;
- # Munge foo/index.html to foo/
- if ($keys[0]=~/(^|.*\/)index.$config{htmlext}$/) {
- # A duplicate might need to be stored under the
- # unmunged name too.
- if (!$config{usedirs} || $config{amazon_s3_dupindex}) {
- push @keys, $1;
- }
- else {
- @keys=($1);
- }
- }
- }
- return @keys;
- }
- package IkiWiki;
- use File::MimeInfo;
- use Encode;
- # This is a wrapper around the real writefile.
- sub writefile ($$$;$$) {
- my $file=shift;
- my $destdir=shift;
- my $content=shift;
- my $binary=shift;
- my $writer=shift;
- # First, write the file to disk.
- my $ret=$IkiWiki::Plugin::amazon_s3::subs{'IkiWiki::writefile'}->($file, $destdir, $content, $binary, $writer);
-
- my @keys=IkiWiki::Plugin::amazon_s3::file2keys("$destdir/$file");
- # Store the data in S3.
- if (@keys) {
- my $bucket=IkiWiki::Plugin::amazon_s3::getbucket();
- # The http layer tries to downgrade utf-8
- # content, but that can fail (see
- # http://rt.cpan.org/Ticket/Display.html?id=35710),
- # so force convert it to bytes.
- $content=encode_utf8($content) if defined $content;
- my %opts=(
- acl_short => 'public-read',
- content_type => mimetype("$destdir/$file"),
- );
- # If there are multiple keys to write, data is sent
- # multiple times.
- # TODO: investigate using the new copy operation.
- # (It may not be robust enough.)
- foreach my $key (@keys) {
- my $res;
- if (! $writer) {
- $res=$bucket->add_key($key, $content, \%opts);
- }
- else {
- # This test for empty files is a workaround
- # for this bug:
- # http://rt.cpan.org//Ticket/Display.html?id=35731
- if (-z "$destdir/$file") {
- $res=$bucket->add_key($key, "", \%opts);
- }
- else {
- # read back in the file that the writer emitted
- $res=$bucket->add_key_filename($key, "$destdir/$file", \%opts);
- }
- }
- if (! $res) {
- error(gettext("Failed to save file to S3: ").
- $bucket->err.": ".$bucket->errstr."\n");
- }
- }
- }
- return $ret;
- }
- # This is a wrapper around the real prune.
- sub prune ($) {
- my $file=shift;
- my @keys=IkiWiki::Plugin::amazon_s3::file2keys($file);
- # Prune files out of S3 too.
- if (@keys) {
- my $bucket=IkiWiki::Plugin::amazon_s3::getbucket();
- foreach my $key (@keys) {
- my $res=$bucket->delete_key($key);
- if (! $res) {
- error(gettext("Failed to delete file from S3: ").
- $bucket->err.": ".$bucket->errstr."\n");
- }
- }
- }
- return $IkiWiki::Plugin::amazon_s3::subs{'IkiWiki::prune'}->($file);
- }
- 1
|