annotate foosdk/sdk/foobar2000/SDK/playlist_loader.cpp @ 1:20d02a178406 default tip

*: check in everything else yay
author Paper <paper@tflc.us>
date Mon, 05 Jan 2026 02:15:46 -0500
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
1 #include "foobar2000-sdk-pch.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
2 #include "playlist_loader.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
3 #include "link_resolver.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
4 #include "archive.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
5 #include "file_info_impl.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
6 #include "input.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
7 #include "advconfig.h"
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
8 #include <string>
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
9 #include <unordered_set>
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
10 #include <list>
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
11
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
12 constexpr unsigned allowRecurseBase = 2; // max. 2 archive levels - mitigate droste.zip stack overflow
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
13 static void process_path_internal(const char * p_path,const service_ptr_t<file> & p_reader,playlist_loader_callback::ptr callback, abort_callback & abort,playlist_loader_callback::t_entry_type type,const t_filestats & p_stats, unsigned allowRecurse );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
14
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
15 bool playlist_loader::g_try_load_playlist(file::ptr fileHint,const char * p_path,playlist_loader_callback::ptr p_callback, abort_callback & p_abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
16 // Determine if this file is a playlist or not (which usually means that it's a media file)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
17 pfc::string8 filepath;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
18
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
19 filesystem::g_get_canonical_path(p_path,filepath);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
20
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
21 pfc::string8 extension = filesystem::g_get_extension(filepath);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
22
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
23 service_ptr_t<file> l_file = fileHint;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
24
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
25 if (l_file.is_empty()) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
26 filesystem::ptr fs;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
27 if (filesystem::g_get_interface(fs,filepath)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
28 if (fs->supports_content_types()) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
29 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
30 fs->open(l_file,filepath,filesystem::open_mode_read,p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
31 } catch(exception_io const &) { return false; } // fall thru
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
32 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
33 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
34 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
35
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
36 service_enum_t<playlist_loader> e;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
37
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
38 if (l_file.is_valid()) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
39
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
40 // Important: in case of remote HTTP files, use actual connected path for matching file extensions, following any redirects.
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
41 // At least one internet radio station has been known to present .pls links that are 302 redirects to real streams, so they don't parse as playlists.
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
42 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
43 file_metadata_http::ptr meta;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
44 if (meta &= l_file->get_metadata_(p_abort)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
45 pfc::string8 realPath;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
46 meta->get_connected_path(realPath);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
47 extension = filesystem::g_get_extension(realPath);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
48 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
49 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
50
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
51 pfc::string8 content_type;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
52 if (l_file->get_content_type(content_type)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
53 for (auto l : e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
54 if (l->is_our_content_type(content_type)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
55 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
56 TRACK_CODE("playlist_loader::open",l->open(filepath,l_file,p_callback, p_abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
57 return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
58 } catch(exception_io_unsupported_format const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
59 l_file->reopen(p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
60 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
61 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
62 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
63 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
64 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
65
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
66 if (extension.length()>0) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
67 for (auto l : e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
68 if (stricmp_utf8(l->get_extension(),extension) == 0) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
69 if (l_file.is_empty()) filesystem::g_open_read(l_file,filepath,p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
70 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
71 TRACK_CODE("playlist_loader::open",l->open(filepath,l_file,p_callback,p_abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
72 return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
73 } catch(exception_io_unsupported_format const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
74 l_file->reopen(p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
75 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
76 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
77 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
78 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
79
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
80 return false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
81 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
82
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
83 void playlist_loader::g_load_playlist_filehint(file::ptr fileHint,const char * p_path,playlist_loader_callback::ptr p_callback, abort_callback & p_abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
84 if (!g_try_load_playlist(fileHint, p_path, p_callback, p_abort)) throw exception_io_unsupported_format();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
85 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
86
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
87 void playlist_loader::g_load_playlist(const char * p_path,playlist_loader_callback::ptr callback, abort_callback & abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
88 g_load_playlist_filehint(NULL,p_path,callback,abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
89 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
90 namespace {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
91 class MIC_impl : public metadb_info_container_v2 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
92 public:
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
93 t_filestats2 const& stats2() override { return m_stats; }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
94 file_info const& info() override { return m_info; }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
95 t_filestats const& stats() override { return m_stats.as_legacy(); }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
96 bool isInfoPartial() override { return false; }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
97
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
98 file_info_impl m_info;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
99 t_filestats2 m_stats;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
100 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
101 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
102 static void index_tracks_helper(const char * p_path,const service_ptr_t<file> & p_reader,const t_filestats & p_stats,playlist_loader_callback::t_entry_type p_type,playlist_loader_callback::ptr p_callback, abort_callback & p_abort,bool & p_got_input)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
103 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
104 TRACK_CALL_TEXT("index_tracks_helper");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
105 if (p_reader.is_empty() && filesystem::g_is_remote_safe(p_path))
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
106 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
107 TRACK_CALL_TEXT("remote");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
108 metadb_handle_ptr handle;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
109 p_callback->handle_create(handle,make_playable_location(p_path,0));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
110 p_got_input = true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
111 p_callback->on_entry(handle,p_type,p_stats,true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
112 } else {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
113 TRACK_CALL_TEXT("hintable");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
114 service_ptr_t<input_info_reader> instance;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
115 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
116 input_entry::g_open_for_info_read(instance,p_reader,p_path,p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
117 } catch(exception_io_unsupported_format const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
118 // specifically bail
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
119 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
120 } catch(exception_io const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
121 // broken file or some other error, open() failed - show it anyway
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
122 metadb_handle_ptr handle;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
123 p_callback->handle_create(handle, make_playable_location(p_path, 0));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
124 p_callback->on_entry(handle, p_type, p_stats, true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
125 return;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
126 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
127
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
128 const auto stats = instance->get_stats2_(p_path, stats2_all, p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
129
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
130 t_uint32 subsong,subsong_count = instance->get_subsong_count();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
131 bool bInfoGetError = false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
132 for(subsong=0;subsong<subsong_count;subsong++)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
133 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
134 TRACK_CALL_TEXT("subsong-loop");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
135 p_abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
136 metadb_handle_ptr handle;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
137 t_uint32 index = instance->get_subsong(subsong);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
138 p_callback->handle_create(handle,make_playable_location(p_path,index));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
139
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
140 p_got_input = true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
141 if (! bInfoGetError && p_callback->want_info(handle,p_type,stats.as_legacy(),true) )
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
142 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
143 auto mic = fb2k::service_new<MIC_impl>();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
144 mic->m_stats = stats;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
145 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
146 TRACK_CODE("get_info",instance->get_info(index,mic->m_info,p_abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
147 } catch(...) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
148 bInfoGetError = true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
149 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
150 if (! bInfoGetError ) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
151 playlist_loader_callback_v2::ptr v2;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
152 if (v2 &= p_callback) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
153 v2->on_entry_info_v2(handle, p_type, mic, true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
154 } else {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
155 p_callback->on_entry_info(handle, p_type, stats.as_legacy(), mic->m_info, true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
156 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
157
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
158 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
159 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
160 else
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
161 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
162 p_callback->on_entry(handle,p_type,stats.as_legacy(),true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
163 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
164 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
165 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
166 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
167
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
168 static void track_indexer__g_get_tracks_wrap(const char * p_path,const service_ptr_t<file> & p_reader,const t_filestats & p_stats,playlist_loader_callback::t_entry_type p_type,playlist_loader_callback::ptr p_callback, abort_callback & p_abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
169 bool got_input = false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
170 bool fail = false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
171 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
172 index_tracks_helper(p_path,p_reader,p_stats,p_type,p_callback,p_abort, got_input);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
173 } catch(exception_aborted const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
174 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
175 } catch(exception_io_unsupported_format const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
176 fail = true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
177 } catch(std::exception const & e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
178 fail = true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
179 FB2K_console_formatter() << "could not enumerate tracks (" << e << ") on:\n" << file_path_display(p_path);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
180 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
181 if (fail) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
182 if (!got_input && !p_abort.is_aborting()) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
183 if (p_type == playlist_loader_callback::entry_user_requested)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
184 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
185 metadb_handle_ptr handle;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
186 p_callback->handle_create(handle,make_playable_location(p_path,0));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
187 p_callback->on_entry(handle,p_type,p_stats,true);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
188 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
189 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
190 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
191 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
192
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
193 namespace {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
194
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
195 static bool queryAddHidden() {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
196 // {2F9F4956-363F-4045-9531-603B1BF39BA8}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
197 static const GUID guid_cfg_addhidden =
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
198 { 0x2f9f4956, 0x363f, 0x4045,{ 0x95, 0x31, 0x60, 0x3b, 0x1b, 0xf3, 0x9b, 0xa8 } };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
199
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
200 advconfig_entry_checkbox::ptr ptr;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
201 if (advconfig_entry::g_find_t(ptr, guid_cfg_addhidden)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
202 return ptr->get_state();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
203 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
204 return false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
205 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
206
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
207 class directory_callback_myimpl : public directory_callback
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
208 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
209 public:
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
210 void main(const char* folder, abort_callback& abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
211 visit(folder);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
212
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
213 abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
214 const uint32_t flags = listMode::filesAndFolders | (queryAddHidden() ? listMode::hidden : 0);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
215
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
216 auto workHere = [&] (folder_t const & f) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
217 filesystem_v2::ptr v2;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
218 if (v2 &= f.m_fs) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
219 v2->list_directory_ex(f.m_folder.c_str(), *this, flags, abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
220 } else {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
221 f.m_fs->list_directory(f.m_folder.c_str(), *this, abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
222 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
223 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
224
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
225 workHere( folder_t { folder, filesystem::get(folder) } );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
226
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
227 for (;; ) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
228 abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
229 auto iter = m_foldersPending.begin();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
230 if ( iter == m_foldersPending.end() ) break;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
231 auto f = std::move(*iter); m_foldersPending.erase(iter);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
232
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
233 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
234 workHere( f );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
235 } catch (exception_io const & e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
236 FB2K_console_formatter() << "Error walking directory (" << e << "): " << f.m_folder.c_str();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
237 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
238 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
239 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
240
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
241 bool on_entry(filesystem * owner,abort_callback & p_abort,const char * url,bool is_subdirectory,const t_filestats & p_stats) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
242 p_abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
243 if (!visit(url)) return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
244
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
245 filesystem_v2::ptr v2;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
246 v2 &= owner;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
247
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
248 if ( is_subdirectory ) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
249 m_foldersPending.emplace_back( folder_t { url, owner } );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
250 } else {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
251 m_entries.emplace_back( entry_t { url, p_stats } );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
252 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
253 return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
254 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
255
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
256 struct entry_t {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
257 std::string m_path;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
258 t_filestats m_stats;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
259 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
260 std::list<entry_t> m_entries;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
261
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
262 bool visit(const char* path) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
263 return m_visited.insert( path ).second;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
264 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
265 std::unordered_set<std::string> m_visited;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
266
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
267 struct folder_t {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
268 std::string m_folder;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
269 filesystem::ptr m_fs;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
270 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
271 std::list<folder_t> m_foldersPending;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
272 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
273 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
274
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
275
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
276 static void process_path_internal(const char * p_path,const service_ptr_t<file> & p_reader,playlist_loader_callback::ptr callback, abort_callback & abort,playlist_loader_callback::t_entry_type type,const t_filestats & p_stats, unsigned allowRecurse)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
277 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
278 if (allowRecurse == 0) return;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
279 //p_path must be canonical
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
280
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
281 abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
282
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
283 callback->on_progress(p_path);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
284
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
285
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
286 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
287 if (p_reader.is_empty() && type != playlist_loader_callback::entry_directory_enumerated) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
288 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
289 directory_callback_myimpl results;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
290 results.main( p_path, abort );
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
291 for( auto & i : results.m_entries ) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
292 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
293 process_path_internal(i.m_path.c_str(), 0, callback, abort, playlist_loader_callback::entry_directory_enumerated, i.m_stats, allowRecurse);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
294 } catch (exception_aborted const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
295 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
296 } catch (std::exception const& e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
297 FB2K_console_formatter() << "Error walking path (" << e << "): " << file_path_display(i.m_path.c_str());
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
298 } catch (...) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
299 FB2K_console_formatter() << "Error walking path (bad exception): " << file_path_display(i.m_path.c_str());
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
300 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
301 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
302 return; // successfully enumerated directory - go no further
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
303 } catch(exception_aborted const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
304 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
305 } catch (exception_io_not_directory const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
306 // disregard
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
307 } catch(exception_io_not_found const &) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
308 // disregard
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
309 } catch (std::exception const& e) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
310 FB2K_console_formatter() << "Error walking directory (" << e << "): " << p_path;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
311 } catch (...) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
312 FB2K_console_formatter() << "Error walking directory (bad exception): " << p_path;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
313 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
314 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
315
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
316 if (allowRecurse > 1) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
317 for (auto f : filesystem::enumerate()) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
318 abort.check();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
319 service_ptr_t<archive> arch;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
320 if ((arch &= f) && arch->is_our_archive(p_path)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
321 if (p_reader.is_valid()) p_reader->reopen(abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
322
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
323 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
324 archive::list_func_t archive_results = [callback, &abort, allowRecurse](const char* p_path, const t_filestats& p_stats, file::ptr p_reader) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
325 process_path_internal(p_path,p_reader,callback,abort,playlist_loader_callback::entry_directory_enumerated,p_stats,allowRecurse - 1);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
326 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
327 TRACK_CODE("archive::archive_list",arch->archive_list(p_path,p_reader,archive_results,/*want readers*/true, abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
328 return;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
329 } catch(exception_aborted const &) {throw;}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
330 catch(...) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
331 // Something failed hard
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
332 // Is is_our_archive() meaningful?
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
333 archive_v2::ptr arch2;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
334 if (arch2 &= arch) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
335 // If yes, show errors
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
336 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
337 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
338 // Outdated archive implementation, preserve legacy behavior (try to open as non-archive)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
339 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
340 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
341 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
342 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
343 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
344
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
345
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
346
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
347 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
348 service_ptr_t<link_resolver> ptr;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
349 if (link_resolver::g_find(ptr,p_path))
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
350 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
351 if (p_reader.is_valid()) p_reader->reopen(abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
352
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
353 pfc::string8 temp;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
354 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
355 TRACK_CODE("link_resolver::resolve",ptr->resolve(p_reader,p_path,temp,abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
356
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
357 track_indexer__g_get_tracks_wrap(temp,0,filestats_invalid,playlist_loader_callback::entry_from_playlist,callback, abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
358 return;//success
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
359 } catch(exception_aborted const &) {throw;}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
360 catch(...) {}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
361 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
362 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
363
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
364 if (callback->is_path_wanted(p_path,type)) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
365 track_indexer__g_get_tracks_wrap(p_path,p_reader,p_stats,type,callback, abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
366 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
367 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
368
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
369 namespace {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
370 class plcallback_simple : public playlist_loader_callback {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
371 public:
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
372 void on_progress(const char* p_path) override { (void)p_path; }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
373
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
374 void on_entry(const metadb_handle_ptr& p_item, t_entry_type p_type, const t_filestats& p_stats, bool p_fresh) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
375 (void)p_type; (void)p_stats; (void)p_fresh;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
376 m_items += p_item;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
377 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
378 bool want_info(const metadb_handle_ptr& p_item, t_entry_type p_type, const t_filestats& p_stats, bool p_fresh) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
379 (void)p_type; (void)p_stats; (void)p_fresh;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
380 return p_item->should_reload(p_stats, p_fresh);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
381 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
382
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
383 void on_entry_info(const metadb_handle_ptr& p_item, t_entry_type p_type, const t_filestats& p_stats, const file_info& p_info, bool p_fresh) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
384 (void)p_type;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
385 m_items += p_item;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
386 m_hints->add_hint(p_item, p_info, p_stats, p_fresh);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
387 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
388
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
389 void handle_create(metadb_handle_ptr& p_out, const playable_location& p_location) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
390 m_metadb->handle_create(p_out, p_location);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
391 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
392
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
393 bool is_path_wanted(const char* path, t_entry_type type) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
394 (void)path; (void)type;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
395 return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
396 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
397
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
398 bool want_browse_info(const metadb_handle_ptr& p_item, t_entry_type p_type, t_filetimestamp ts) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
399 (void)p_item; (void)p_type; (void)ts;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
400 return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
401 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
402 void on_browse_info(const metadb_handle_ptr& p_item, t_entry_type p_type, const file_info& info, t_filetimestamp ts) override {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
403 (void)p_type;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
404 metadb_hint_list_v2::ptr v2;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
405 if (v2 &= m_hints) v2->add_hint_browse(p_item, info, ts);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
406 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
407
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
408 ~plcallback_simple() {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
409 m_hints->on_done();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
410 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
411 metadb_handle_list m_items;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
412 private:
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
413 const metadb_hint_list::ptr m_hints = metadb_hint_list::create();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
414 const metadb::ptr m_metadb = metadb::get();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
415 };
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
416 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
417 void playlist_loader::g_path_to_handles_simple(const char* p_path, pfc::list_base_t<metadb_handle_ptr>& p_out, abort_callback& p_abort) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
418 auto cb = fb2k::service_new<plcallback_simple>();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
419 g_process_path(p_path, cb, p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
420 p_out = cb->m_items;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
421 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
422 void playlist_loader::g_process_path(const char * p_filename,playlist_loader_callback::ptr callback, abort_callback & abort,playlist_loader_callback::t_entry_type type)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
423 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
424 TRACK_CALL_TEXT("playlist_loader::g_process_path");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
425
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
426 auto filename = file_path_canonical(p_filename);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
427
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
428 process_path_internal(filename,0,callback,abort, type,filestats_invalid, allowRecurseBase);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
429 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
430
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
431 void playlist_loader::g_save_playlist(const char * p_filename,const pfc::list_base_const_t<metadb_handle_ptr> & data,abort_callback & p_abort)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
432 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
433 TRACK_CALL_TEXT("playlist_loader::g_save_playlist");
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
434 pfc::string8 filename;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
435 filesystem::g_get_canonical_path(p_filename,filename);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
436 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
437 service_ptr_t<file> r;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
438 filesystem::g_open(r,filename,filesystem::open_mode_write_new,p_abort);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
439
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
440 auto ext = pfc::string_extension(filename);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
441
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
442 service_enum_t<playlist_loader> e;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
443 service_ptr_t<playlist_loader> l;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
444 if (e.first(l)) do {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
445 if (l->can_write() && !stricmp_utf8(ext,l->get_extension())) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
446 try {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
447 TRACK_CODE("playlist_loader::write",l->write(filename,r,data,p_abort));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
448 return;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
449 } catch(exception_io_data const &) {}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
450 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
451 } while(e.next(l));
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
452 throw exception_io_data();
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
453 } catch(...) {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
454 try {filesystem::g_remove(filename,p_abort);} catch(...) {}
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
455 throw;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
456 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
457 }
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
458
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
459
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
460 bool playlist_loader::g_process_path_ex(const char * filename,playlist_loader_callback::ptr callback, abort_callback & abort,playlist_loader_callback::t_entry_type type)
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
461 {
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
462 if (g_try_load_playlist(NULL, filename, callback, abort)) return true;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
463 //not a playlist format
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
464 g_process_path(filename,callback,abort,type);
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
465 return false;
20d02a178406 *: check in everything else
Paper <paper@tflc.us>
parents:
diff changeset
466 }