# ==================================================================
# Gossamer Links - enhanced directory management system
#
# Website : http://gossamer-threads.com/
# Support : http://gossamer-threads.com/scripts/support/
# CVS Info : 087,071,086,086,085
# Revision : $Id: Admin.pm,v 1.16 2005/03/05 01:29:09 brewt Exp $
#
# Copyright (c) 2001 Gossamer Threads Inc. All Rights Reserved.
# Redistribution in part or in whole strictly prohibited. Please
# see LICENSE file for full details.
# ==================================================================
package Links::Admin;
# ==================================================================
use strict;
use GT::SQL::Admin;
use Links qw/$DB/;
use GT::AutoLoader;
use vars qw/@ISA $ERROR_MESSAGE $FONT $DEBUG/;
@ISA = qw/GT::SQL::Admin/;
$DEBUG = 0;
$ERROR_MESSAGE = 'GT::SQL';
$FONT = $GT::SQL::Admin::FONT;
# Make sure AUTOLOAD does not catch destroyed objects.
sub DESTROY {}
$COMPILE{modify_multi_records} = __LINE__ . <<'END_OF_SUB';
sub modify_multi_records {
# -------------------------------------------------------------------
# Overrides the Links table to format the category name properly.
#
my $self = shift;
my $name = $self->{table}->name;
my $prefix = $DB->prefix;
return $self->SUPER::modify_multi_records(@_) unless ( $name eq $prefix . 'Links');
if (! exists $self->{cgi}->{modify}) {
return $self->modify_error("Please select a record to modify before continuing.");
}
# If they selected only one record to modify we still need an array ref
ref $self->{cgi}->{modify} eq 'ARRAY' or $self->{cgi}->{modify} = [$self->{cgi}->{modify}];
# Format the cgi for inserting
$self->format_insert_cgi;
# Hash to handle errors if there are any errors.
my $errors = {};
my $errcode = {};
# Need to know the names of the columns for this Table.
my @columns = keys %{$self->{table}->cols};
push @columns, 'CatLinks.CategoryID';
# Need to know the number of records modified
my $rec_modified = 0;
# For through the record numbers. These are the values of the
# check boxes
foreach my $rec_num (@{$self->{cgi}->{modify}}) {
my $change = {};
foreach my $column (@columns) {
$change->{$column} = $self->{cgi}->{"$rec_num-$column"} if exists $self->{cgi}->{"$rec_num-$column"};
}
# Make the changes and capture any errors.
my $ret = $self->{table}->modify($change);
if (defined $ret) {
$rec_modified++;
}
else {
if ($self->{table}->error) {
my $error = $self->{table}->error;
$error =~ s/\n/
\n
You can either import from a file or you can cut and paste the contents into a textarea box. If you
have a large number of records, you should really import from a file. The first row of your input should be the
fully qualified column names. You must also include the Category ID and Category Name of the category the link
will be imported to. |
Import data from file: or from textarea box: |
"; print $self->_end_form; print $self->_prop_navbar; print "
";
print $self->_footer;
print $self->_end_html;
}
END_OF_SUB
$COMPILE{editor_import_data} = __LINE__ . <<'END_OF_SUB';
sub editor_import_data {
# -------------------------------------------------------------------
# Allow the import to import category/link data. Only used if called with the
# Links database.
#
my $self = shift;
my $name = $self->{table}->name;
my $prefix = $DB->prefix;
return $self->SUPER::editor_import_data(@_) unless ( $name eq $prefix . 'Links');
my $delim = $self->{cgi}->{'import-delim'} or return $self->editor_import_data_form ("No import delimiter specified!");
my $file = $self->{cgi}->{'import-file'};
my $text = $self->{cgi}->{'import-text'};
# Make sure there is some data to import
$file or $text or return $self->editor_import_data_form ("You must enter in at least a filename or data in the textarea box.");
$file and $text and return $self->editor_import_data_form ("Please only enter either a filename or data in the textarea box, not both.");
$delim =~ s/\\t/\t/g;
$delim =~ /%/ and $self->editor_import_data_form("% may not be used as a delimited.");
# Store the lines to import in @lines and the header in $header.
my ($good_cnt, $err_cnt, $line, $line_num, @lines, @data, $error, %record, $i);
if ($file) {
local *FILE;
open (FILE, "<$file") or return $self->editor_import_data_form ("Unable to open file: '$file'. Reason: $!");
local $/;
@lines = split /[\r\n]+/, scalar From here you can export your Links to either the screen or to
a file on your server. The first line of the export will be a list of the column
headers. The last two fields is the Category ID the link is in, and the Category Name.
If a link is in more then one category, you will get one row for each occurrence.
";
print $self->_end_form;
print $self->_prop_navbar;
print " ";
print $self->_footer;
print $self->_end_html;
}
END_OF_SUB
$COMPILE{editor_export_data} = __LINE__ . <<'END_OF_SUB';
sub editor_export_data {
# -------------------------------------------------------------------
# Allow the export to export category/link data. Only used if called with the
# Links database.
#
my $self = shift;
my $name = $self->{table}->name;
my $prefix = $DB->prefix;
return $self->SUPER::editor_export_data(@_) unless ( $name eq $prefix . 'Links');
print $self->{in}->header;
ref $self->{cgi}->{db} and return $self->error('BADARGS','FATAL', "Editor can only be called with one table, not a relation.");
my @order = $self->{table}->ordered_columns;
@order or return $self->editor_export_data_form("No fields selected to export (@order).");
# Add on the prefix.
for (@order) { $_ = $prefix . 'Links.' . $_; }
# Add the ID and Category Name.
push @order, $prefix . 'Category.ID', $prefix .'Category.Name';
my $delim = $self->{cgi}->{'export-delim'};
$delim = "\t" if $delim eq '\t';
length $delim or $self->editor_export_data_form("No delimiter entered.");
$delim =~ /%/ and $self->editor_export_data_form("% may not be used as a delimited.");
my $screen = $self->{cgi}->{'export-mode'} ne 'file';
local *FILE;
if ($screen) {
open FILE, ">&STDOUT";
print FILE $self->{in}->header(); # print FILE to avoid STDOUT vs. FILE buffering issues
print FILE " Warning: deleting a user will also delete all links associated with that user! Warning: deleting a category will also delete all sub categories and links in those categories!$GT::SQL::error
\n";
$err_cnt++;
next LINE;
}
$link_map{$link_id} = $link_id;
}
else {
my $old_id = $link_id;
my $sth = $Links->insert(\%record);
unless ($sth and ($link_id = $sth->insert_id)) {
$error .= "$GT::SQL::error
\n";
$err_cnt++;
next LINE;
}
$link_map{$old_id} = $link_id;
}
}
else {
my $old_id = $link_id;
my $sth = $Links->insert(\%record);
unless ($sth and ($link_id = $sth->insert_id)) {
$error .= "$GT::SQL::error
\n";
$err_cnt++;
next LINE;
}
$link_map{$old_id} = $link_id;
}
}
unless ($CatLinks->count ( { LinkID => $link_id, CategoryID => $cat_id })) {
$CatLinks->insert({ LinkID => $link_id, CategoryID => $cat_id });
}
$good_cnt++;
last if ($err_cnt > 100);
}
# Return the results.
if ($error) {
return $self->editor_import_data_form (($err_cnt >= 100) ?
"Aborting, too many errors!
Rows imported: $good_cnt
Errors with the following rows:
$error
" :
"Rows imported: $good_cnt
Errors with the following rows: $error
");
}
return $self->editor_import_data_form ("Rows imported: $good_cnt.");
}
END_OF_SUB
$COMPILE{editor_export_data_form} = __LINE__ . <<'END_OF_SUB';
sub editor_export_data_form {
# -------------------------------------------------------------------
# Allow the export to export category/link data. Only used if called with the
# Links database.
#
my $self = shift;
my $name = $self->{table}->name;
my $prefix = $DB->prefix;
return $self->SUPER::editor_export_data_form(@_) unless ( $name eq $prefix . 'Links');
my $msg = shift;
print $self->{in}->header;
$msg &&= qq|$msg|;
my $table = $self->{record};
print $self->_start_html ( { title => "Table Editor: $table" });
print $self->_header ("Table Editor", $msg || "Export Data from $table.");
print $self->_start_form ( { do => 'editor_export_data', db => $self->{cgi}->{db} }, {name => 'ExportForm'});
print qq~
Export data to:
filename:
Use as delimiter.
~;
print $self->_buttons ("Export Data from");
print "";
}
else {
my $filename = $self->{cgi}->{'export-file'} or return $self->editor_export_data_form("Please enter a file name!");
open FILE, "> $filename" or return $self->editor_export_data_form("Unable to open file '$filename': $!");
}
# Print the row header.
print FILE join ($delim, @order), "\n";
# Print the data.
my $db = $DB->table(qw/Links CatLinks Category/);
my $sth = $db->select(\@order) or return $self->editor_export_data_form($GT::SQL::error);
my $delim_re = quotemeta $delim;
my $delim_str = join '', map sprintf("%%%02x", ord($_)), split '', $delim;
{
local $, = $delim;
local $\ = "\n";
while (my $row = $sth->fetchrow_arrayref) {
for (@$row) {
s{$delim_re}{$delim_str}g;
s{%} {%25}g;
s{\r} {}g;
s{\n} {%0a}g;
$_ = $self->{in}->html_escape($_) if $screen;
}
print FILE @$row;
}
}
print FILE "
" if $screen;
return $self->editor_export_data_form("Data has been exported to: $self->{cgi}->{'export-file'}") unless $screen;
return;
}
END_OF_SUB
sub _check_opts {
# -------------------------------------------------------------------
# Need to override this so searching for categories works.
#
my $self = shift;
my $sel = 0;
# Relation does not plat fare :(
my $cols = $self->{table}->cols;
for (keys %{$self->{cgi}}) { $sel = 1 if (($self->{cgi}->{$_} =~ /\S/) and exists $cols->{$_}) }
if ((exists $self->{cgi}->{query} and $self->{cgi}->{query} =~ /\S/) or
(exists $self->{cgi}->{keyword} and $self->{cgi}->{keyword} =~ /\S/)) {
$sel = 1;
}
my $prefix = $DB->prefix;
if (! $sel and ($self->{table}->name eq $prefix . 'Links') and (exists $self->{cgi}->{'CatLinks.CategoryID'})) {
$sel = 1;
}
$sel or return;
return 1;
}
sub _buttons {
# -------------------------------------------------------------------
# Adds a warning message to delete Users and Categories.
#
my $self = shift;
my $name = shift;
my $prefix = GT::SQL->prefix();
my $msg = '';
if (($self->{table}->name eq $prefix . "Users") and ($name eq 'Delete')) {
$msg = qq~
~;
}
1;
$msg