33 :
Thread (
"DownloadTask thread"),
40 jassert (fileStream !=
nullptr);
41 jassert (stream !=
nullptr);
43 targetLocation = fileStream->getFile();
44 contentLength = stream->getTotalLength();
45 httpCode = stream->getStatusCode();
62 if (listener !=
nullptr)
63 listener->
progress (
this, downloaded, contentLength);
65 auto max = jmin ((
int) bufferSize, contentLength < 0 ? std::numeric_limits<int>::max()
66 :
static_cast<int> (contentLength - downloaded));
68 auto actual = stream->read (buffer.
get(), max);
73 if (! fileStream->write (buffer.
get(),
static_cast<size_t> (
actual)))
81 if (downloaded == contentLength)
90 if (contentLength > 0 && downloaded < contentLength)
100 const std::unique_ptr<FileOutputStream> fileStream;
101 const std::unique_ptr<WebInputStream> stream;
102 const size_t bufferSize;
110URL::DownloadTask::Listener::~Listener() {}
119 const size_t bufferSize = 0x8000;
127 if (stream->connect (
nullptr))
134URL::DownloadTask::DownloadTask() {}
175 url =
"file://" + url;
196 else if (nextAmp > 0 && equalsPos < nextAmp)
199 equalsPos < 0 ? String() :
removeEscapeChars (url.substring (equalsPos + 1, nextAmp)));
210URL::URL (
const String& u,
int) : url (u) {}
213 : url (std::move (other.url)),
214 postData (std::move (other.postData)),
215 parameterNames (std::move (other.parameterNames)),
216 parameterValues (std::move (other.parameterValues)),
217 filesToUpload (std::move (other.filesToUpload))
219 , bookmark (std::move (other.bookmark))
224URL& URL::operator= (URL&& other)
226 url = std::move (other.url);
227 postData = std::move (other.postData);
228 parameterNames = std::move (other.parameterNames);
229 parameterValues = std::move (other.parameterValues);
230 filesToUpload = std::move (other.filesToUpload);
232 bookmark = std::move (other.bookmark);
247 return url ==
other.url
248 && postData ==
other.postData
249 && parameterNames ==
other.parameterNames
250 && parameterValues ==
other.parameterValues
251 && filesToUpload ==
other.filesToUpload;
254bool URL::operator!= (
const URL&
other)
const
261 static String getMangledParameters (
const URL& url)
263 jassert (url.getParameterNames().size() == url.getParameterValues().size());
266 for (
int i = 0; i < url.getParameterNames().size(); ++i)
271 auto val = url.getParameterValues()[i];
275 if (val.isNotEmpty())
282 static int findEndOfScheme (
const String& url)
287 || url[i] ==
'+' || url[i] ==
'-' || url[i] ==
'.')
290 return url.substring (i).startsWith (
"://") ? i + 1 : 0;
293 static int findStartOfNetLocation (
const String& url)
295 int start = findEndOfScheme (url);
297 while (url[start] ==
'/')
303 static int findStartOfPath (
const String& url)
305 return url.indexOfChar (findStartOfNetLocation (url),
'/') + 1;
308 static void concatenatePaths (String& path,
const String& suffix)
310 if (! path.endsWithChar (
'/'))
313 if (suffix.startsWithChar (
'/'))
314 path += suffix.substring (1);
320void URL::addParameter (
const String& name,
const String& value)
322 parameterNames.
add (name);
323 parameterValues.
add (value);
329 return url +
"?" + URLHelpers::getMangledParameters (*
this);
342 return url.isNotEmpty();
347 return getDomainInternal (
false);
352 auto startOfPath = URLHelpers::findStartOfPath (url);
360 return url.
substring (0, URLHelpers::findEndOfScheme (url) - 1);
371 return fileFromFileSchemeURL (*
this);
380File URL::fileFromFileSchemeURL (
const URL& fileURL)
382 if (! fileURL.isLocalFile())
388 auto path =
removeEscapeChars (fileURL.getDomainInternal (
true)).replace (
"+",
"%2B");
391 bool isUncPath = (! fileURL.url.startsWith (
"file:///"));
398 for (
auto urlElement : urlElements)
399 path += File::getSeparatorString() +
removeEscapeChars (urlElement.replace (
"+",
"%2B"));
403 path =
"\\\\" + path;
411 auto colonPos = url.indexOfChar (URLHelpers::findStartOfNetLocation (url),
':');
425 const int startOfPath = URLHelpers::findStartOfPath (url);
432 URLHelpers::concatenatePaths (u.url,
newPath);
439 URLHelpers::concatenatePaths (u.url,
subPath);
447 if (filesToUpload.
size() > 0)
450 jassert (postData.
getSize() == 0);
454 headers <<
"Content-Type: multipart/form-data; boundary=" <<
boundary <<
"\r\n";
458 for (
int i = 0; i < parameterNames.
size(); ++i)
460 data <<
"\r\nContent-Disposition: form-data; name=\"" << parameterNames[i]
461 <<
"\"\r\n\r\n" << parameterValues[i]
465 for (
auto* f : filesToUpload)
467 data <<
"\r\nContent-Disposition: form-data; name=\"" << f->parameterName
468 <<
"\"; filename=\"" << f->filename <<
"\"\r\n";
470 if (f->mimeType.isNotEmpty())
471 data <<
"Content-Type: " << f->mimeType <<
"\r\n";
473 data <<
"Content-Transfer-Encoding: binary\r\n\r\n";
475 if (f->data !=
nullptr)
480 data <<
"\r\n--" << boundary;
487 data << URLHelpers::getMangledParameters (*
this)
492 headers <<
"Content-Type: application/x-www-form-urlencoded\r\n";
494 headers <<
"Content-length: " << (int) data.getDataSize() <<
"\r\n";
501 static const char*
validProtocols[] = {
"http:",
"ftp:",
"https:" };
512 .fromLastOccurrenceOf (
".",
false,
false));
528 auto start = URLHelpers::findStartOfNetLocation (url);
529 auto end1 = url.indexOfChar (start,
'/');
535 return url.substring (start, end);
539URL::Bookmark::Bookmark (
void* bookmarkToUse)
540 : data (bookmarkToUse)
544URL::Bookmark::~Bookmark()
546 [(NSData*) data release];
549void setURLBookmark (URL& u,
void* bookmark)
551 u.bookmark =
new URL::Bookmark (bookmark);
554void* getURLBookmark (URL& u)
556 if (u.bookmark.get() ==
nullptr)
559 return u.bookmark.get()->
data;
565template <
typename Stream>
570 : Stream (getLocalFileAccess (
urlToUse)),
592 updateStaleBookmark (
nsURL, url);
607 bool securityAccessSucceeded =
false;
652 setURLBookmark (
juceUrl, (
void*) bookmark);
718 if (statusCode !=
nullptr)
719 *statusCode =
wi->getStatusCode();
747 return juce_CreateContentURIOutputStream (*
this);
774 return in->readEntireStreamAsString();
818 : parameterName (
param), filename (name), mimeType (
mime), file (f), data (
mb)
820 jassert (mimeType.isNotEmpty());
823URL URL::withUpload (Upload*
const f)
const
827 for (
int i = u.filesToUpload.size(); --i >= 0;)
828 if (u.filesToUpload.getObjectPointerUnchecked(i)->parameterName == f->parameterName)
829 u.filesToUpload.remove (i);
831 u.filesToUpload.add (f);
836 const String& mimeType)
const
838 return withUpload (
new Upload (parameterName,
fileToUpload.getFileName(),
845 return withUpload (
new Upload (parameterName, filename, mimeType,
File(),
852 auto result =
s.replaceCharacter (
'+',
' ');
854 if (! result.containsChar (
'%'))
859 Array<char> utf8 (result.toRawUTF8(), (
int) result.getNumBytesAsUTF8());
861 for (
int i = 0; i <
utf8.
size(); ++i)
889 for (
int i = 0; i <
utf8.
size(); ++i)
910 if (u.containsChar (
'@') && ! u.containsChar (
':'))
Holds a resizable array of primitive or copy-by-value objects.
ElementType getUnchecked(int index) const
Returns one of the elements in the array, without checking the index passed in.
bool isEmpty() const noexcept
Returns true if the array is empty, false otherwise.
int size() const noexcept
Returns the current number of elements in the array.
void removeRange(int startIndex, int numberToRemove)
Removes a range of elements from the array.
void insert(int indexToInsertAt, ParameterType newElement)
Inserts a new element into the array at a given position.
ElementType * getRawDataPointer() noexcept
Returns a pointer to the actual array data.
void set(int indexToChange, ParameterType newValue)
Replaces an element with a new value.
ElementType * data() const noexcept
Returns a pointer to the first element in the array.
static int getHexDigitValue(juce_wchar digit) noexcept
Returns 0 to 16 for '0' to 'F", or -1 for characters that aren't a legal hex digit.
static bool isLetterOrDigit(char character) noexcept
Checks whether a character is alphabetic or numeric.
An output stream that writes into a local file.
Represents a local file or directory.
FileInputStream * createInputStream() const
Creates a stream to read from this file.
static StringRef getSeparatorString()
The system-specific file separator character, as a string.
ElementType * get() const noexcept
Returns a raw pointer to the allocated data.
A class to hold a resizable block of raw data.
size_t getSize() const noexcept
Returns the block's current allocated size, in bytes.
Writes data to an internal memory buffer, which grows as required.
The base class for streams that write data to some kind of destination.
static bool JUCE_CALLTYPE openDocument(const String &documentURL, const String ¶meters)
Tries to launch the OS's default reader application for a given file or URL.
static Random & getSystemRandom() noexcept
The overhead of creating a new Random object is fairly small, but if you want to avoid it,...
int size() const noexcept
Returns the current number of objects in the array.
static StringArray fromTokens(StringRef stringToTokenise, bool preserveQuotedStrings)
Returns an array containing the tokens in a given string.
int size() const noexcept
Returns the number of strings in the array.
void add(String stringToAdd)
Appends a string at the end of the array.
A container for holding a set of strings which are keyed by another string.
int indexOfChar(juce_wchar characterToLookFor) const noexcept
Searches for a character inside this string.
String upToFirstOccurrenceOf(StringRef substringToEndWith, bool includeSubStringInResult, bool ignoreCase) const
Returns the start of this string, up to the first occurrence of a substring.
bool containsIgnoreCase(StringRef text) const noexcept
Tests whether the string contains another substring.
bool startsWithChar(juce_wchar character) const noexcept
Tests whether the string begins with a particular character.
static String toHexString(IntegerType number)
Returns a string representing this numeric value in hexadecimal.
String substring(int startIndex, int endIndex) const
Returns a subsection of the string.
String fromLastOccurrenceOf(StringRef substringToFind, bool includeSubStringInResult, bool ignoreCase) const
Returns a section of the string starting from the last occurrence of a given substring.
static String fromUTF8(const char *utf8buffer, int bufferSizeBytes=-1)
Creates a String from a UTF-8 encoded buffer.
String fromFirstOccurrenceOf(StringRef substringToStartFrom, bool includeSubStringInResult, bool ignoreCase) const
Returns a section of the string starting from a given substring.
void startThread()
Starts the thread running.
bool waitForThreadToExit(int timeOutMilliseconds) const
Waits for the thread to stop.
bool threadShouldExit() const
Checks whether the thread has been told to stop running.
void signalThreadShouldExit()
Sets a flag to tell the thread it should stop.
Represents a download task.
virtual ~DownloadTask()
Releases the resources of the download task, unregisters the listener and cancels the download if nec...
Represents a URL and has a bunch of useful functions to manipulate it.
URL() noexcept
Creates an empty URL.
URL withParameter(const String ¶meterName, const String ¶meterValue) const
Returns a copy of this URL, with a GET or POST parameter added to the end.
static URL createWithoutParsing(const String &url)
Returns a URL without attempting to remove any embedded parameters from the string.
File getLocalFile() const
Returns the file path of the local file to which this URL refers to.
InputStream * createInputStream(bool doPostLikeRequest, OpenStreamProgressCallback *progressCallback=nullptr, void *progressCallbackContext=nullptr, String extraHeaders=String(), int connectionTimeOutMs=0, StringPairArray *responseHeaders=nullptr, int *statusCode=nullptr, int numRedirectsToFollow=5, String httpRequestCmd=String()) const
Attempts to open a stream that can read from this URL.
bool isWellFormed() const
True if it seems to be valid.
bool readEntireBinaryStream(MemoryBlock &destData, bool usePostCommand=false) const
Tries to download the entire contents of this URL into a binary data block.
int getPort() const
Attempts to read a port number from the URL.
XmlElement * readEntireXmlStream(bool usePostCommand=false) const
Tries to download the entire contents of this URL and parse it as XML.
OutputStream * createOutputStream() const
Attempts to open an output stream to a URL for writing.
URL withDataToUpload(const String ¶meterName, const String &filename, const MemoryBlock &fileContentToUpload, const String &mimeType) const
Returns a copy of this URL, with a file-upload type parameter added to it.
String getFileName() const
Returns the file name.
URL getChildURL(const String &subPath) const
Returns a new URL that refers to a sub-path relative to this one.
static String removeEscapeChars(const String &stringToRemoveEscapeCharsFrom)
Replaces any escape character sequences in a string with their original character codes.
static String addEscapeChars(const String &stringToAddEscapeCharsTo, bool isParameter, bool roundBracketsAreLegal=true)
Adds escape sequences to a string to encode any characters that aren't legal in a URL.
String toString(bool includeGetParameters) const
Returns a string version of the URL.
String getSubPath() const
Returns the path part of the URL.
URL withNewSubPath(const String &newPath) const
Returns a new version of this URL with a different sub-path.
static bool isProbablyAnEmailAddress(const String &possibleEmailAddress)
Takes a guess as to whether a string might be a valid email address.
URL withNewDomainAndPath(const String &newFullPath) const
Returns a new version of this URL with a different domain and path.
String readEntireTextStream(bool usePostCommand=false) const
Tries to download the entire contents of this URL as a string.
bool isEmpty() const noexcept
Returns true if the URL is an empty string.
static bool isProbablyAWebsiteURL(const String &possibleURL)
Takes a guess as to whether a string might be a valid website address.
bool(void *context, int bytesSent, int totalBytes) OpenStreamProgressCallback
This callback function can be used by the createInputStream() method.
String getDomain() const
Returns just the domain part of the URL.
bool isLocalFile() const
Returns true if this URL refers to a local file.
URL withFileToUpload(const String ¶meterName, const File &fileToUpload, const String &mimeType) const
Returns a copy of this URL, with a file-upload type parameter added to it.
URL withPOSTData(const String &postData) const
Returns a copy of this URL, with a block of data to send as the POST data.
URL withParameters(const StringPairArray ¶metersToAdd) const
Returns a copy of this URL, with a set of GET or POST parameters added.
String getScheme() const
Returns the scheme of the URL.
bool operator==(const URL &) const
Compares two URLs.
bool launchInDefaultBrowser() const
Tries to launch the system's default browser to open the URL.
static XmlElement * parse(const File &file)
A handy static method that parses a file.
Used to build a tree of elements representing an XML document.
void run() override
Must be implemented to perform the thread's actual code.
Used to receive callbacks for download progress.
virtual void finished(URL::DownloadTask *task, bool success)=0
Called when the download has finished.
virtual void progress(URL::DownloadTask *task, int64 bytesDownloaded, int64 totalLength)
Called periodically by the OS to indicate download progress.