mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-25 10:00:23 -04:00 
			
		
		
		
	Samples are downloaded from a web server, currently the SF download server. The samples are stored in the source controlled samples directory and the CMake script there builds a suitable directory tree for upload to the web server under samples/web containing the samples hierarchy and the generated JSON contents database file. The samples CMake script also defines an 'upload-samples' target that uses rsync to efficiently upload the samples and the accompanying contents JSON database file. Any directory structure under the samples directory may be created, to add a new sample file simply add the file to source control and amend the list of sample files (SAMPLE_FILES) in samples/CMakeLists.txt. git-svn-id: svn+ssh://svn.code.sf.net/p/wsjt/wsjt/branches/wsjtx@6308 ab8295b8-cf94-4d9e-aec4-7959e3be5d79
		
			
				
	
	
		
			300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "Directory.hpp"
 | |
| 
 | |
| #include <QVariant>
 | |
| #include <QString>
 | |
| #include <QHeaderView>
 | |
| #include <QStringList>
 | |
| #include <QFileInfo>
 | |
| #include <QDir>
 | |
| #include <QNetworkAccessManager>
 | |
| #include <QAuthenticator>
 | |
| #include <QNetworkReply>
 | |
| #include <QTreeWidgetItem>
 | |
| #include <QTreeWidgetItemIterator>
 | |
| #include <QMessageBox>
 | |
| #include <QJsonDocument>
 | |
| #include <QJsonParseError>
 | |
| #include <QJsonArray>
 | |
| #include <QJsonObject>
 | |
| #include <QRegularExpression>
 | |
| 
 | |
| #include "Configuration.hpp"
 | |
| #include "DirectoryNode.hpp"
 | |
| #include "FileNode.hpp"
 | |
| #include "revision_utils.hpp"
 | |
| 
 | |
| #include "moc_Directory.cpp"
 | |
| 
 | |
| namespace
 | |
| {
 | |
|   char const * samples_dir_name = "samples";
 | |
|   QString const contents_file_name = "contents_" + version (false) + ".json";
 | |
| }
 | |
| 
 | |
| Directory::Directory (Configuration const * configuration
 | |
|                       , QNetworkAccessManager * network_manager
 | |
|                       , QWidget * parent)
 | |
|   : QTreeWidget {parent}
 | |
|   , configuration_ {configuration}
 | |
|   , network_manager_ {network_manager}
 | |
|   , root_dir_ {configuration_->save_directory ()}
 | |
|   , contents_ {this
 | |
|         , network_manager_
 | |
|         , QDir {root_dir_.absoluteFilePath (samples_dir_name)}.absoluteFilePath (contents_file_name)}
 | |
| {
 | |
|   dir_icon_.addPixmap (style ()->standardPixmap (QStyle::SP_DirClosedIcon), QIcon::Normal, QIcon::Off);
 | |
|   dir_icon_.addPixmap (style ()->standardPixmap (QStyle::SP_DirOpenIcon), QIcon::Normal, QIcon::On);
 | |
|   file_icon_.addPixmap (style ()->standardPixmap (QStyle::SP_FileIcon));
 | |
| 
 | |
|   setColumnCount (2);
 | |
|   setHeaderLabels ({"File", "Progress"});
 | |
|   header ()->setSectionResizeMode (QHeaderView::ResizeToContents);
 | |
|   setItemDelegate (&item_delegate_);
 | |
| 
 | |
|   connect (network_manager_, &QNetworkAccessManager::authenticationRequired
 | |
|            , this, &Directory::authentication);
 | |
|   connect (this, &Directory::itemChanged, [this] (QTreeWidgetItem * item) {
 | |
|       switch (item->type ())
 | |
|         {
 | |
|         case FileNode::Type:
 | |
|           {
 | |
|             FileNode * node = static_cast<FileNode *> (item);
 | |
|             if (!node->sync (node->checkState (0) == Qt::Checked))
 | |
|               {
 | |
|                 FileNode::sync_blocker b {node};
 | |
|                 node->setCheckState (0, node->checkState (0) == Qt::Checked ? Qt::Unchecked : Qt::Checked);
 | |
|               }
 | |
|           }
 | |
|           break;
 | |
| 
 | |
|         default:
 | |
|           break;
 | |
|         }
 | |
|     });
 | |
| }
 | |
| 
 | |
| bool Directory::url_root (QUrl root)
 | |
| {
 | |
|   if (!root.path ().endsWith ('/'))
 | |
|     {
 | |
|       root.setPath (root.path () + '/');
 | |
|     }
 | |
|   if (root.isValid ())
 | |
|     {
 | |
|       url_root_ = root;
 | |
|       refresh ();
 | |
|     }
 | |
|   return root.isValid ();
 | |
| }
 | |
| 
 | |
| void Directory::error (QString const& title, QString const& message)
 | |
| {
 | |
|   QMessageBox::warning (this, title, message);
 | |
| }
 | |
| 
 | |
| bool Directory::refresh ()
 | |
| {
 | |
|   abort ();
 | |
|   clear ();
 | |
|   // update locations
 | |
|   root_dir_ = configuration_->save_directory ();
 | |
|   QDir contents_dir {root_dir_.absoluteFilePath (samples_dir_name)};
 | |
|   contents_.local_file_path (contents_dir.absoluteFilePath (contents_file_name));
 | |
|   QUrl url {url_root_.resolved (QDir {root_dir_.relativeFilePath (samples_dir_name)}.filePath (contents_file_name))};
 | |
|   if (url.isValid ())
 | |
|     {
 | |
|       return contents_.sync (url, true, true); // attempt to fetch contents
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       QMessageBox::warning (this
 | |
|                             , tr ("URL Error")
 | |
|                             , tr ("Invalid URL:\n\"%1\"")
 | |
|                             .arg (url.toDisplayString ()));
 | |
|     }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void Directory::download_finished (bool success)
 | |
| {
 | |
|   if (success)
 | |
|     {
 | |
|       QFile contents {contents_.local_file_path ()};
 | |
|       if (contents.open (QFile::ReadOnly | QFile::Text))
 | |
|         {
 | |
|           QJsonParseError json_status;
 | |
|           auto content = QJsonDocument::fromJson (contents.readAll (), &json_status);
 | |
|           if (json_status.error)
 | |
|             {
 | |
|               QMessageBox::warning (this
 | |
|                                     , tr ("JSON Error")
 | |
|                                     , tr ("Contents file syntax error %1 at character offset %2")
 | |
|                                     .arg (json_status.errorString ()).arg (json_status.offset));
 | |
|               return;
 | |
|             }
 | |
|           if (!content.isArray ())
 | |
|             {
 | |
|               QMessageBox::warning (this, tr ("JSON Error")
 | |
|                                     , tr ("Contents file top level must be a JSON array"));
 | |
|               return;
 | |
|             }
 | |
|           QTreeWidgetItem * parent {invisibleRootItem ()};
 | |
|           parent = new DirectoryNode {parent, samples_dir_name};
 | |
|           parent->setIcon (0, dir_icon_);
 | |
|           parent->setExpanded (true);
 | |
|           parse_entries (content.array (), root_dir_.relativeFilePath (samples_dir_name), parent);
 | |
|         }
 | |
|       else
 | |
|         {
 | |
|           QMessageBox::warning (this, tr ("File System Error")
 | |
|                                 , tr ("Failed to open \"%1\"\nError: %2 - %3")
 | |
|                                 .arg (contents.fileName ())
 | |
|                                 .arg (contents.error ())
 | |
|                                 .arg (contents.errorString ()));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Directory::parse_entries (QJsonArray const& entries, QDir const& dir, QTreeWidgetItem * parent)
 | |
| {
 | |
|   if (dir.isRelative () && !dir.path ().startsWith ('.'))
 | |
|     {
 | |
|       for (auto const& value: entries)
 | |
|         {
 | |
|           if (value.isObject ())
 | |
|             {
 | |
|               auto const& entry = value.toObject ();
 | |
|               auto const& name = entry["name"].toString ();
 | |
|               if (name.size () && !name.contains (QRegularExpression {R"([/:;])"}))
 | |
|                 {
 | |
|                   auto const& type = entry["type"].toString ();
 | |
|                   if ("file" == type)
 | |
|                     {
 | |
|                       QUrl url {url_root_.resolved (dir.filePath (name))};
 | |
|                       if (url.isValid ())
 | |
|                         {
 | |
|                           auto node = new FileNode {parent, network_manager_
 | |
|                                                     , QDir {root_dir_.filePath (dir.path ())}.absoluteFilePath (name)
 | |
|                                                     , url};
 | |
|                           FileNode::sync_blocker b {node};
 | |
|                           node->setIcon (0, file_icon_);
 | |
|                           node->setCheckState (0, node->local () ? Qt::Checked : Qt::Unchecked);
 | |
|                           update (parent);
 | |
|                         }
 | |
|                       else
 | |
|                         {
 | |
|                           QMessageBox::warning (this
 | |
|                                                 , tr ("URL Error")
 | |
|                                                 , tr ("Invalid URL:\n\"%1\"")
 | |
|                                                 .arg (url.toDisplayString ()));
 | |
|                         }
 | |
|                     }
 | |
|                   else if ("directory" == type)
 | |
|                     {
 | |
|                           auto node = new DirectoryNode {parent, name};
 | |
|                           node->setIcon (0, dir_icon_);
 | |
|                           auto const& entries = entry["entries"];
 | |
|                           if (entries.isArray ())
 | |
|                             {
 | |
|                               parse_entries (entries.toArray ()
 | |
|                                              , QDir {root_dir_.relativeFilePath (dir.path ())}.filePath (name)
 | |
|                                              , node);
 | |
|                             }
 | |
|                           else
 | |
|                             {
 | |
|                               QMessageBox::warning (this, tr ("JSON Error")
 | |
|                                                     , tr ("Contents entries must be a JSON array"));
 | |
|                             }
 | |
|                     }
 | |
|                   else
 | |
|                     {
 | |
|                       QMessageBox::warning (this, tr ("JSON Error")
 | |
|                                             , tr ("Contents entries must have a valid type"));
 | |
|                     }
 | |
|                 }
 | |
|               else
 | |
|                 {
 | |
|                   QMessageBox::warning (this, tr ("JSON Error")
 | |
|                                         , tr ("Contents entries must have a valid name"));
 | |
|                 }
 | |
|             }
 | |
|           else
 | |
|             {
 | |
|               QMessageBox::warning (this, tr ("JSON Error")
 | |
|                                     , tr ("Contents entries must be JSON objects"));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       QMessageBox::warning (this, tr ("JSON Error")
 | |
|                             , tr ("Contents directories must be relative and within \"%1\"")
 | |
|                             .arg (samples_dir_name));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Directory::abort ()
 | |
| {
 | |
|   QTreeWidgetItemIterator iter {this};
 | |
|   while (*iter)
 | |
|     {
 | |
|       if ((*iter)->type () == FileNode::Type)
 | |
|         {
 | |
|           auto * node = static_cast<FileNode *> (*iter);
 | |
|           node->abort ();
 | |
|         }
 | |
|       ++iter;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Directory::update (QTreeWidgetItem * item)
 | |
| {
 | |
|   // iterate the tree under item and accumulate the progress
 | |
|   if (item)
 | |
|     {
 | |
|       Q_ASSERT (item->type () == DirectoryNode::Type);
 | |
|       qint64 max {0};
 | |
|       qint64 bytes {0};
 | |
| 
 | |
|       // reset progress
 | |
|       item->setData (1, Qt::UserRole, max);
 | |
|       item->setData (1, Qt::DisplayRole, bytes);
 | |
|       int items {0};
 | |
|       int counted {0};
 | |
|       QTreeWidgetItemIterator iter {item};
 | |
|       // iterate sub tree only
 | |
|       while (*iter && (*iter == item || (*iter)->parent () != item->parent ()))
 | |
|         {
 | |
|           if ((*iter)->type () == FileNode::Type)  // only count files
 | |
|             {
 | |
|               ++items;
 | |
|               if (auto size = (*iter)->data (1, Qt::UserRole).toLongLong ())
 | |
|                 {
 | |
|                   max += size;
 | |
|                   ++counted;
 | |
|                 }
 | |
|               bytes += (*iter)->data (1, Qt::DisplayRole).toLongLong ();
 | |
|             }
 | |
|           ++iter;
 | |
|         }
 | |
|       // estimate size of items not yet downloaded as average of
 | |
|       // those actually present
 | |
|       if (counted)
 | |
|         {
 | |
|           max += (items - counted) * max / counted;
 | |
|         }
 | |
|       // save as our progress
 | |
|       item->setData (1, Qt::UserRole, max);
 | |
|       item->setData (1, Qt::DisplayRole, bytes);
 | |
| 
 | |
|       // recurse up to top
 | |
|       update (item->parent ());
 | |
|     }
 | |
| }
 | |
| 
 | |
| void Directory::authentication (QNetworkReply * /* reply */
 | |
|                                 , QAuthenticator * /* authenticator */)
 | |
| {
 | |
|   QMessageBox::warning (this, "Network Error", "Authentication required");
 | |
| }
 |