xref: /plugin/struct/admin/schemas.php (revision ad86a8246773ce555271b9f3e75eb4f5fbf66227)
187fdbc6bSMichael Große<?php
2d6d97f60SAnna Dabrowska
387fdbc6bSMichael Große/**
487fdbc6bSMichael Große * DokuWiki Plugin struct (Admin Component)
587fdbc6bSMichael Große *
687fdbc6bSMichael Große * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html
787fdbc6bSMichael Große * @author  Andreas Gohr, Michael Große <dokuwiki@cosmocode.de>
887fdbc6bSMichael Große */
987fdbc6bSMichael Große
1087fdbc6bSMichael Großeuse dokuwiki\Form\Form;
11f36cc634SAndreas Gohruse dokuwiki\plugin\struct\meta\CSVExporter;
1269d92b7aSAnna Dabrowskause dokuwiki\plugin\struct\meta\CSVImporter;
131fc2361fSSzymon Olewniczakuse dokuwiki\plugin\struct\meta\CSVPageImporter;
14*ad86a824SAnna Dabrowskause dokuwiki\plugin\struct\meta\CSVSerialImporter;
15ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\Schema;
16ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\SchemaBuilder;
17ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\SchemaEditor;
18ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\SchemaImporter;
19ba766201SAndreas Gohruse dokuwiki\plugin\struct\meta\StructException;
2087fdbc6bSMichael Große
21d6d97f60SAnna Dabrowskaclass admin_plugin_struct_schemas extends DokuWiki_Admin_Plugin
22d6d97f60SAnna Dabrowska{
2387fdbc6bSMichael Große
2487fdbc6bSMichael Große    /**
2587fdbc6bSMichael Große     * @return int sort number in admin menu
2687fdbc6bSMichael Große     */
27d6d97f60SAnna Dabrowska    public function getMenuSort()
28d6d97f60SAnna Dabrowska    {
2987fdbc6bSMichael Große        return 500;
3087fdbc6bSMichael Große    }
3187fdbc6bSMichael Große
3287fdbc6bSMichael Große    /**
3387fdbc6bSMichael Große     * @return bool true if only access for superuser, false is for superusers and moderators
3487fdbc6bSMichael Große     */
35d6d97f60SAnna Dabrowska    public function forAdminOnly()
36d6d97f60SAnna Dabrowska    {
374d220607SAndreas Gohr        return false;
3887fdbc6bSMichael Große    }
3987fdbc6bSMichael Große
4087fdbc6bSMichael Große    /**
4187fdbc6bSMichael Große     * Should carry out any processing required by the plugin.
4287fdbc6bSMichael Große     */
43d6d97f60SAnna Dabrowska    public function handle()
44d6d97f60SAnna Dabrowska    {
4587fdbc6bSMichael Große        global $INPUT;
46d5a1a6dcSAndreas Gohr        global $ID;
47e33460e2SMichael Grosse        global $config_cascade;
48e33460e2SMichael Grosse        $config_file_path = end($config_cascade['main']['local']);
4987fdbc6bSMichael Große
508ddf87afSAndreas Gohr        // form submit
5187fdbc6bSMichael Große        $table = Schema::cleanTableName($INPUT->str('table'));
5287fdbc6bSMichael Große        if ($table && $INPUT->bool('save') && checkSecurityToken()) {
53d5a1a6dcSAndreas Gohr            $builder = new SchemaBuilder($table, $INPUT->arr('schema'));
5487fdbc6bSMichael Große            if (!$builder->build()) {
5587fdbc6bSMichael Große                msg('something went wrong while saving', -1);
5687fdbc6bSMichael Große            }
57ae5c46faSAndreas Gohr            touch(action_plugin_struct_cache::getSchemaRefreshFile());
5887fdbc6bSMichael Große        }
598ddf87afSAndreas Gohr        // export
60d486d6d7SAndreas Gohr        if ($table && $INPUT->bool('export')) {
61d5a1a6dcSAndreas Gohr            $builder = new Schema($table);
62d486d6d7SAndreas Gohr            header('Content-Type: application/json');
63d486d6d7SAndreas Gohr            header("Content-Disposition: attachment; filename=$table.struct.json");
64d486d6d7SAndreas Gohr            echo $builder->toJSON();
65d486d6d7SAndreas Gohr            exit;
66d486d6d7SAndreas Gohr        }
678ddf87afSAndreas Gohr        // import
688ddf87afSAndreas Gohr        if ($table && $INPUT->bool('import')) {
698ddf87afSAndreas Gohr            if (isset($_FILES['schemafile']['tmp_name'])) {
708ddf87afSAndreas Gohr                $json = io_readFile($_FILES['schemafile']['tmp_name'], false);
718ddf87afSAndreas Gohr                if (!$json) {
728ddf87afSAndreas Gohr                    msg('Something went wrong with the upload', -1);
738ddf87afSAndreas Gohr                } else {
7469d92b7aSAnna Dabrowska                    $builder = new SchemaImporter($table, $json);
758ddf87afSAndreas Gohr                    if (!$builder->build()) {
768ddf87afSAndreas Gohr                        msg('something went wrong while saving', -1);
778ddf87afSAndreas Gohr                    }
78ae5c46faSAndreas Gohr                    touch(action_plugin_struct_cache::getSchemaRefreshFile());
798ddf87afSAndreas Gohr                }
808ddf87afSAndreas Gohr            }
818ddf87afSAndreas Gohr        }
82a0b3799eSAndreas Gohr
83a0b3799eSAndreas Gohr        // import CSV
84a0b3799eSAndreas Gohr        if ($table && $INPUT->bool('importcsv')) {
85a0b3799eSAndreas Gohr            if (isset($_FILES['csvfile']['tmp_name'])) {
86a0b3799eSAndreas Gohr                try {
8728d21c10SAnna Dabrowska                    $datatype = $INPUT->str('importtype');
88*ad86a824SAnna Dabrowska                    if ($datatype === CSVExporter::DATATYPE_PAGE) {
8928d21c10SAnna Dabrowska                        $csvImporter = new CSVPageImporter($table, $_FILES['csvfile']['tmp_name'], $datatype);
90*ad86a824SAnna Dabrowska                    } else if ($datatype === CSVExporter::DATATYPE_SERIAL) {
91*ad86a824SAnna Dabrowska                        $csvImporter = new CSVSerialImporter($table, $_FILES['csvfile']['tmp_name'], $datatype);
9269d92b7aSAnna Dabrowska                    } else {
9328d21c10SAnna Dabrowska                        $csvImporter = new CSVImporter($table, $_FILES['csvfile']['tmp_name'], $datatype);
941fc2361fSSzymon Olewniczak                    }
951fc2361fSSzymon Olewniczak                    $csvImporter->import();
966d2df532SAndreas Gohr                    msg($this->getLang('admin_csvdone'), 1);
97a0b3799eSAndreas Gohr                } catch (StructException $e) {
98a0b3799eSAndreas Gohr                    msg(hsc($e->getMessage()), -1);
99a0b3799eSAndreas Gohr                }
100a0b3799eSAndreas Gohr            }
101a0b3799eSAndreas Gohr        }
102a0b3799eSAndreas Gohr
103f36cc634SAndreas Gohr        // export CSV
104f36cc634SAndreas Gohr        if ($table && $INPUT->bool('exportcsv')) {
105f36cc634SAndreas Gohr            header('Content-Type: text/csv');
106f36cc634SAndreas Gohr            header('Content-Disposition: attachment; filename="' . $table . '.csv";');
10769d92b7aSAnna Dabrowska            new CSVExporter($table, $INPUT->str('exporttype'));
108f36cc634SAndreas Gohr            exit();
109f36cc634SAndreas Gohr        }
110f36cc634SAndreas Gohr
111d5a1a6dcSAndreas Gohr        // delete
112d5a1a6dcSAndreas Gohr        if ($table && $INPUT->bool('delete')) {
113d5a1a6dcSAndreas Gohr            if ($table != $INPUT->str('confirm')) {
114d5a1a6dcSAndreas Gohr                msg($this->getLang('del_fail'), -1);
115d5a1a6dcSAndreas Gohr            } else {
116d5a1a6dcSAndreas Gohr                try {
117d5a1a6dcSAndreas Gohr                    $schema = new Schema($table);
118d5a1a6dcSAndreas Gohr                    $schema->delete();
119d5a1a6dcSAndreas Gohr                    msg($this->getLang('del_ok'), 1);
120ae5c46faSAndreas Gohr                    touch(action_plugin_struct_cache::getSchemaRefreshFile());
121d5a1a6dcSAndreas Gohr                    send_redirect(wl($ID, array('do' => 'admin', 'page' => 'struct_schemas'), true, '&'));
122d5a1a6dcSAndreas Gohr                } catch (StructException $e) {
123d5a1a6dcSAndreas Gohr                    msg(hsc($e->getMessage()), -1);
124d5a1a6dcSAndreas Gohr                }
125d5a1a6dcSAndreas Gohr            }
126d5a1a6dcSAndreas Gohr        }
127d5a1a6dcSAndreas Gohr
12879c83e06SMichael Große        // clear
12979c83e06SMichael Große        if ($table && $INPUT->bool('clear')) {
13079c83e06SMichael Große            if ($table != $INPUT->str('confirm_clear')) {
13179c83e06SMichael Große                msg($this->getLang('clear_fail'), -1);
13279c83e06SMichael Große            } else {
13379c83e06SMichael Große                try {
13479c83e06SMichael Große                    $schema = new Schema($table);
13579c83e06SMichael Große                    $schema->clear();
13679c83e06SMichael Große                    msg($this->getLang('clear_ok'), 1);
13779c83e06SMichael Große                    touch(action_plugin_struct_cache::getSchemaRefreshFile());
13879c83e06SMichael Große                    send_redirect(wl($ID, array('do' => 'admin', 'page' => 'struct_schemas'), true, '&'));
13979c83e06SMichael Große                } catch (StructException $e) {
14079c83e06SMichael Große                    msg(hsc($e->getMessage()), -1);
14179c83e06SMichael Große                }
14279c83e06SMichael Große            }
14379c83e06SMichael Große        }
14487fdbc6bSMichael Große    }
14587fdbc6bSMichael Große
14687fdbc6bSMichael Große    /**
14787fdbc6bSMichael Große     * Render HTML output, e.g. helpful text and a form
14887fdbc6bSMichael Große     */
149d6d97f60SAnna Dabrowska    public function html()
150d6d97f60SAnna Dabrowska    {
15187fdbc6bSMichael Große        global $INPUT;
15287fdbc6bSMichael Große
15387fdbc6bSMichael Große        $table = Schema::cleanTableName($INPUT->str('table'));
15487fdbc6bSMichael Große        if ($table) {
1550ceefd5cSAnna Dabrowska            $schema = new Schema($table, 0);
1567c080d69SAndreas Gohr
1576af24d3eSAndreas Gohr            echo $this->locale_xhtml('editor_edit');
1580ceefd5cSAnna Dabrowska            echo '<h2>' . sprintf($this->getLang('edithl'), hsc($table)) . '</h2>';
1598ddf87afSAndreas Gohr
1608ddf87afSAndreas Gohr            echo '<ul class="tabs" id="plugin__struct_tabs">';
1618ddf87afSAndreas Gohr            /** @noinspection HtmlUnknownAnchorTarget */
1628ddf87afSAndreas Gohr            echo '<li class="active"><a href="#plugin__struct_editor">' . $this->getLang('tab_edit') . '</a></li>';
1638ddf87afSAndreas Gohr            /** @noinspection HtmlUnknownAnchorTarget */
1648ddf87afSAndreas Gohr            echo '<li><a href="#plugin__struct_json">' . $this->getLang('tab_export') . '</a></li>';
165d5a1a6dcSAndreas Gohr            /** @noinspection HtmlUnknownAnchorTarget */
166d5a1a6dcSAndreas Gohr            echo '<li><a href="#plugin__struct_delete">' . $this->getLang('tab_delete') . '</a></li>';
1678ddf87afSAndreas Gohr            echo '</ul>';
1688ddf87afSAndreas Gohr            echo '<div class="panelHeader"></div>';
1698ddf87afSAndreas Gohr
1707c080d69SAndreas Gohr            $editor = new SchemaEditor($schema);
17187fdbc6bSMichael Große            echo $editor->getEditor();
172748e747fSAnna Dabrowska            echo $this->htmlJson($schema);
173748e747fSAnna Dabrowska            echo $this->htmlDelete($schema);
17487fdbc6bSMichael Große        } else {
1756af24d3eSAndreas Gohr            echo $this->locale_xhtml('editor_intro');
176748e747fSAnna Dabrowska            echo $this->htmlNewschema();
17787fdbc6bSMichael Große        }
17887fdbc6bSMichael Große    }
17987fdbc6bSMichael Große
18087fdbc6bSMichael Große    /**
18169d92b7aSAnna Dabrowska     * Form for handling import/export from/to JSON and CSV
1820845722bSAndreas Gohr     *
1830845722bSAndreas Gohr     * @param Schema $schema
1848ddf87afSAndreas Gohr     * @return string
1858ddf87afSAndreas Gohr     */
186748e747fSAnna Dabrowska    protected function htmlJson(Schema $schema)
187d6d97f60SAnna Dabrowska    {
1888ddf87afSAndreas Gohr        $form = new Form(array('enctype' => 'multipart/form-data', 'id' => 'plugin__struct_json'));
1898ddf87afSAndreas Gohr        $form->setHiddenField('do', 'admin');
1908ddf87afSAndreas Gohr        $form->setHiddenField('page', 'struct_schemas');
1910845722bSAndreas Gohr        $form->setHiddenField('table', $schema->getTable());
1928ddf87afSAndreas Gohr
19369d92b7aSAnna Dabrowska        // schemas
1948ddf87afSAndreas Gohr        $form->addFieldsetOpen($this->getLang('export'));
1958ddf87afSAndreas Gohr        $form->addButton('export', $this->getLang('btn_export'));
1968ddf87afSAndreas Gohr        $form->addFieldsetClose();
1978ddf87afSAndreas Gohr
1988ddf87afSAndreas Gohr        $form->addFieldsetOpen($this->getLang('import'));
1990bd068beSMichael Große        $form->addElement(new \dokuwiki\Form\InputElement('file', 'schemafile'))->attr('accept', '.json');
2008ddf87afSAndreas Gohr        $form->addButton('import', $this->getLang('btn_import'));
2018ddf87afSAndreas Gohr        $form->addHTML('<p>' . $this->getLang('import_warning') . '</p>');
2028ddf87afSAndreas Gohr        $form->addFieldsetClose();
203a0b3799eSAndreas Gohr
20469d92b7aSAnna Dabrowska        // data
2056d2df532SAndreas Gohr        $form->addFieldsetOpen($this->getLang('admin_csvexport'));
20669d92b7aSAnna Dabrowska        $form->addTagOpen('legend');
20769d92b7aSAnna Dabrowska        $form->addHTML($this->getLang('admin_csvexport_datatype'));
20869d92b7aSAnna Dabrowska        $form->addTagClose('legend');
2094fc908c2SAnna Dabrowska        $form->addRadioButton('exporttype', $this->getLang('admin_csv_page'))
2104fc908c2SAnna Dabrowska            ->val('page')
2114fc908c2SAnna Dabrowska            ->attr('checked', 'checked')->addClass('edit block');
2124fc908c2SAnna Dabrowska        $form->addRadioButton('exporttype', $this->getLang('admin_csv_lookup'))
2134fc908c2SAnna Dabrowska            ->val('lookup')
2144fc908c2SAnna Dabrowska            ->addClass('edit block');
2154fc908c2SAnna Dabrowska        $form->addRadioButton('exporttype', $this->getLang('admin_csv_serial'))
2164fc908c2SAnna Dabrowska            ->val('serial')
2174fc908c2SAnna Dabrowska            ->addClass('edit block');
21869d92b7aSAnna Dabrowska        $form->addHTML('<br>');
219f36cc634SAndreas Gohr        $form->addButton('exportcsv', $this->getLang('btn_export'));
220f36cc634SAndreas Gohr        $form->addFieldsetClose();
221f36cc634SAndreas Gohr
2226d2df532SAndreas Gohr        $form->addFieldsetOpen($this->getLang('admin_csvimport'));
22369d92b7aSAnna Dabrowska        $form->addTagOpen('legend');
22469d92b7aSAnna Dabrowska        $form->addHTML($this->getLang('admin_csvimport_datatype'));
22569d92b7aSAnna Dabrowska        $form->addTagClose('legend');
2264fc908c2SAnna Dabrowska        $form->addRadioButton('importtype', $this->getLang('admin_csv_page'))
2274fc908c2SAnna Dabrowska            ->val('page')
2284fc908c2SAnna Dabrowska            ->attr('checked', 'checked')
2294fc908c2SAnna Dabrowska            ->addClass('edit block');
2304fc908c2SAnna Dabrowska        $form->addRadioButton('importtype', $this->getLang('admin_csv_lookup'))
2314fc908c2SAnna Dabrowska            ->val('lookup')
2324fc908c2SAnna Dabrowska            ->addClass('edit block');
2334fc908c2SAnna Dabrowska        $form->addRadioButton('importtype', $this->getLang('admin_csv_serial'))
2344fc908c2SAnna Dabrowska            ->val('serial')
2354fc908c2SAnna Dabrowska            ->addClass('edit block');
23669d92b7aSAnna Dabrowska        $form->addHTML('<br>');
2370bd068beSMichael Große        $form->addElement(new \dokuwiki\Form\InputElement('file', 'csvfile'))->attr('accept', '.csv');
238a0b3799eSAndreas Gohr        $form->addButton('importcsv', $this->getLang('btn_import'));
239a22f2592SMichael Große        $form->addCheckbox('createPage', 'Create missing pages')->addClass('block edit');
2406d2df532SAndreas Gohr        $form->addHTML('<p><a href="https://www.dokuwiki.org/plugin:struct:csvimport">' . $this->getLang('admin_csvhelp') . '</a></p>');
241a0b3799eSAndreas Gohr        $form->addFieldsetClose();
242a0b3799eSAndreas Gohr
2438ddf87afSAndreas Gohr        return $form->toHTML();
2448ddf87afSAndreas Gohr    }
2458ddf87afSAndreas Gohr
2468ddf87afSAndreas Gohr    /**
247d5a1a6dcSAndreas Gohr     * Form for deleting schemas
2480845722bSAndreas Gohr     *
2490845722bSAndreas Gohr     * @param Schema $schema
250d5a1a6dcSAndreas Gohr     * @return string
251d5a1a6dcSAndreas Gohr     */
252748e747fSAnna Dabrowska    protected function htmlDelete(Schema $schema)
253d6d97f60SAnna Dabrowska    {
254d5a1a6dcSAndreas Gohr        $form = new Form(array('id' => 'plugin__struct_delete'));
255d5a1a6dcSAndreas Gohr        $form->setHiddenField('do', 'admin');
256d5a1a6dcSAndreas Gohr        $form->setHiddenField('page', 'struct_schemas');
2570845722bSAndreas Gohr        $form->setHiddenField('table', $schema->getTable());
258d5a1a6dcSAndreas Gohr
25979c83e06SMichael Große        $form->addFieldsetOpen($this->getLang('btn_delete'));
260d5a1a6dcSAndreas Gohr        $form->addHTML($this->locale_xhtml('delete_intro'));
261d5a1a6dcSAndreas Gohr        $form->addTextInput('confirm', $this->getLang('del_confirm'));
262d5a1a6dcSAndreas Gohr        $form->addButton('delete', $this->getLang('btn_delete'));
263d5a1a6dcSAndreas Gohr        $form->addFieldsetClose();
26479c83e06SMichael Große
26579c83e06SMichael Große        $form->addFieldsetOpen($this->getLang('btn_clear'));
26679c83e06SMichael Große        $form->addHTML($this->locale_xhtml('clear_intro'));
26779c83e06SMichael Große        $form->addTextInput('confirm_clear', $this->getLang('clear_confirm'));
26879c83e06SMichael Große        $form->addButton('clear', $this->getLang('btn_clear'));
26979c83e06SMichael Große        $form->addFieldsetClose();
27079c83e06SMichael Große
271d5a1a6dcSAndreas Gohr        return $form->toHTML();
272d5a1a6dcSAndreas Gohr    }
273d5a1a6dcSAndreas Gohr
274d5a1a6dcSAndreas Gohr    /**
27587fdbc6bSMichael Große     * Form to add a new schema
2768ddf87afSAndreas Gohr     *
2778ddf87afSAndreas Gohr     * @return string
27887fdbc6bSMichael Große     */
279748e747fSAnna Dabrowska    protected function htmlNewschema()
280d6d97f60SAnna Dabrowska    {
28187fdbc6bSMichael Große        $form = new Form();
2824e427bd5SAndreas Gohr        $form->addClass('struct_newschema');
28387fdbc6bSMichael Große        $form->addFieldsetOpen($this->getLang('create'));
28487fdbc6bSMichael Große        $form->setHiddenField('do', 'admin');
285dbffe06eSAndreas Gohr        $form->setHiddenField('page', 'struct_schemas');
28687fdbc6bSMichael Große        $form->addTextInput('table', $this->getLang('schemaname'));
28787fdbc6bSMichael Große        $form->addButton('', $this->getLang('save'));
28887fdbc6bSMichael Große        $form->addHTML('<p>' . $this->getLang('createhint') . '</p>'); // FIXME is that true? we probably could
28987fdbc6bSMichael Große        $form->addFieldsetClose();
2908ddf87afSAndreas Gohr        return $form->toHTML();
29187fdbc6bSMichael Große    }
29287fdbc6bSMichael Große
29387fdbc6bSMichael Große    /**
29487fdbc6bSMichael Große     * Adds all available schemas to the Table of Contents
29587fdbc6bSMichael Große     *
29687fdbc6bSMichael Große     * @return array
29787fdbc6bSMichael Große     */
298d6d97f60SAnna Dabrowska    public function getTOC()
299d6d97f60SAnna Dabrowska    {
30087fdbc6bSMichael Große        global $ID;
30187fdbc6bSMichael Große
30287fdbc6bSMichael Große        $toc = array();
3038ddf87afSAndreas Gohr        $link = wl(
304d6d97f60SAnna Dabrowska            $ID,
305d6d97f60SAnna Dabrowska            array(
30687fdbc6bSMichael Große                   'do' => 'admin',
307dbffe06eSAndreas Gohr                   'page' => 'struct_assignments'
3088ddf87afSAndreas Gohr               )
3098ddf87afSAndreas Gohr        );
310dbffe06eSAndreas Gohr        $toc[] = html_mktocitem($link, $this->getLang('menu_assignments'), 0, '');
3117c080d69SAndreas Gohr        $slink = wl(
312d6d97f60SAnna Dabrowska            $ID,
313d6d97f60SAnna Dabrowska            array(
314dbffe06eSAndreas Gohr                   'do' => 'admin',
315dbffe06eSAndreas Gohr                   'page' => 'struct_schemas'
3168ddf87afSAndreas Gohr               )
3178ddf87afSAndreas Gohr        );
3187c080d69SAndreas Gohr        $toc[] = html_mktocitem($slink, $this->getLang('menu'), 0, '');
31987fdbc6bSMichael Große
3200ceefd5cSAnna Dabrowska        $tables = Schema::getAll();
3217c080d69SAndreas Gohr        if ($tables) {
322097f4a53SAndreas Gohr            foreach ($tables as $table) {
3238ddf87afSAndreas Gohr                $link = wl(
324d6d97f60SAnna Dabrowska                    $ID,
325d6d97f60SAnna Dabrowska                    array(
32687fdbc6bSMichael Große                           'do' => 'admin',
327dbffe06eSAndreas Gohr                           'page' => 'struct_schemas',
328097f4a53SAndreas Gohr                           'table' => $table
3298ddf87afSAndreas Gohr                       )
3308ddf87afSAndreas Gohr                );
33187fdbc6bSMichael Große
3320ceefd5cSAnna Dabrowska                $toc[] = html_mktocitem($link, hsc($table), 1, '');
3337c080d69SAndreas Gohr            }
3347c080d69SAndreas Gohr        }
3357c080d69SAndreas Gohr
33687fdbc6bSMichael Große        return $toc;
33787fdbc6bSMichael Große    }
33887fdbc6bSMichael Große}
33987fdbc6bSMichael Große
34087fdbc6bSMichael Große// vim:ts=4:sw=4:et:
341