2018-03-26 14:32:43 -07:00
|
|
|
// @flow
|
2018-10-29 13:23:53 -04:00
|
|
|
import * as MODALS from 'constants/modal_types';
|
2019-07-12 10:58:24 -04:00
|
|
|
import * as ACTIONS from 'constants/action_types';
|
2019-10-23 15:39:51 -04:00
|
|
|
import * as PAGES from 'constants/pages';
|
2021-10-17 16:36:14 +08:00
|
|
|
import { batchActions } from 'util/batch-actions';
|
|
|
|
import { doCheckPendingClaims } from 'redux/actions/claims';
|
2020-04-24 09:51:00 -04:00
|
|
|
import {
|
2021-03-29 19:05:18 -04:00
|
|
|
makeSelectClaimForUri,
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
selectMyActiveClaims,
|
2021-10-17 16:36:14 +08:00
|
|
|
selectMyClaims,
|
|
|
|
selectMyChannelClaims,
|
|
|
|
// selectMyClaimsWithoutChannels,
|
|
|
|
selectReflectingById,
|
|
|
|
} from 'redux/selectors/claims';
|
|
|
|
import { makeSelectPublishFormValue, selectPublishFormValues, selectMyClaimForUri } from 'redux/selectors/publish';
|
2020-06-12 16:44:25 -04:00
|
|
|
import { doError } from 'redux/actions/notifications';
|
2019-07-12 22:59:45 -04:00
|
|
|
import { push } from 'connected-react-router';
|
|
|
|
import analytics from 'analytics';
|
2020-10-05 14:31:51 -04:00
|
|
|
import { doOpenModal } from 'redux/actions/app';
|
2021-10-17 16:36:14 +08:00
|
|
|
import { CC_LICENSES, COPYRIGHT, OTHER, NONE, PUBLIC_DOMAIN } from 'constants/licenses';
|
2021-12-16 16:16:22 +08:00
|
|
|
import { SPEECH_STATUS, SPEECH_PUBLISH } from 'constants/speech_urls';
|
2021-10-17 16:36:14 +08:00
|
|
|
import * as THUMBNAIL_STATUSES from 'constants/thumbnail_upload_statuses';
|
|
|
|
import { creditsToString } from 'util/format-credits';
|
|
|
|
import Lbry from 'lbry';
|
|
|
|
// import LbryFirst from 'extras/lbry-first/lbry-first';
|
|
|
|
import { isClaimNsfw } from 'util/claim';
|
2021-12-23 12:21:25 -06:00
|
|
|
import { LBRY_FIRST_TAG, SCHEDULED_LIVESTREAM_TAG } from 'constants/tags';
|
2021-10-17 16:36:14 +08:00
|
|
|
|
2021-10-21 23:17:17 +08:00
|
|
|
function resolveClaimTypeForAnalytics(claim) {
|
|
|
|
if (!claim) {
|
|
|
|
return 'undefined_claim';
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (claim.value_type) {
|
|
|
|
case 'stream':
|
|
|
|
if (claim.value) {
|
|
|
|
if (!claim.value.source) {
|
|
|
|
return 'livestream';
|
|
|
|
} else {
|
|
|
|
return claim.value.stream_type;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return 'stream';
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// collection, channel, repost, undefined
|
|
|
|
return claim.value_type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-26 18:03:16 -04:00
|
|
|
export const NO_FILE = '---';
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
|
|
|
|
function resolvePublishPayload(publishData, myClaimForUri, myChannels, preview) {
|
|
|
|
const {
|
|
|
|
name,
|
|
|
|
bid,
|
|
|
|
filePath,
|
|
|
|
description,
|
|
|
|
language,
|
|
|
|
releaseTimeEdited,
|
2021-12-30 13:36:28 -06:00
|
|
|
releaseAnytime,
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
// license,
|
|
|
|
licenseUrl,
|
|
|
|
useLBRYUploader,
|
|
|
|
licenseType,
|
|
|
|
otherLicenseDescription,
|
|
|
|
thumbnail,
|
|
|
|
channel,
|
|
|
|
title,
|
|
|
|
contentIsFree,
|
|
|
|
fee,
|
|
|
|
// uri,
|
|
|
|
tags,
|
|
|
|
// locations,
|
|
|
|
optimize,
|
|
|
|
isLivestreamPublish,
|
|
|
|
remoteFileUrl,
|
|
|
|
} = publishData;
|
|
|
|
|
|
|
|
// Handle scenario where we have a claim that has the same name as a channel we are publishing with.
|
|
|
|
const myClaimForUriEditing = myClaimForUri && myClaimForUri.name === name ? myClaimForUri : null;
|
|
|
|
|
|
|
|
let publishingLicense;
|
|
|
|
switch (licenseType) {
|
|
|
|
case COPYRIGHT:
|
|
|
|
case OTHER:
|
|
|
|
publishingLicense = otherLicenseDescription;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
publishingLicense = licenseType;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the claim id from the channel name, we will use that instead
|
|
|
|
const namedChannelClaim = myChannels ? myChannels.find((myChannel) => myChannel.name === channel) : null;
|
|
|
|
const channelId = namedChannelClaim ? namedChannelClaim.claim_id : '';
|
|
|
|
|
2021-12-30 13:36:28 -06:00
|
|
|
const nowTimeStamp = Number(Math.round(Date.now() / 1000));
|
|
|
|
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
const publishPayload: {
|
|
|
|
name: ?string,
|
|
|
|
bid: string,
|
|
|
|
description?: string,
|
|
|
|
channel_id?: string,
|
|
|
|
file_path?: string,
|
|
|
|
license_url?: string,
|
|
|
|
license?: string,
|
|
|
|
thumbnail_url?: string,
|
2021-12-30 13:36:28 -06:00
|
|
|
release_time: number,
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
fee_currency?: string,
|
|
|
|
fee_amount?: string,
|
|
|
|
languages?: Array<string>,
|
|
|
|
tags: Array<string>,
|
|
|
|
locations?: Array<any>,
|
|
|
|
blocking: boolean,
|
|
|
|
optimize_file?: boolean,
|
|
|
|
preview?: boolean,
|
|
|
|
remote_url?: string,
|
|
|
|
} = {
|
|
|
|
name,
|
|
|
|
title,
|
|
|
|
description,
|
|
|
|
locations: [],
|
|
|
|
bid: creditsToString(bid),
|
|
|
|
languages: [language],
|
|
|
|
tags: tags && tags.map((tag) => tag.name),
|
|
|
|
thumbnail_url: thumbnail,
|
2021-12-30 13:36:28 -06:00
|
|
|
release_time: nowTimeStamp,
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
blocking: true,
|
|
|
|
preview: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Temporary solution to keep the same publish flow with the new tags api
|
|
|
|
// Eventually we will allow users to enter their own tags on publish
|
|
|
|
// `nsfw` will probably be removed
|
|
|
|
if (remoteFileUrl) {
|
|
|
|
publishPayload.remote_url = remoteFileUrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (publishingLicense) {
|
|
|
|
publishPayload.license = publishingLicense;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (licenseUrl) {
|
|
|
|
publishPayload.license_url = licenseUrl;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (thumbnail) {
|
|
|
|
publishPayload.thumbnail_url = thumbnail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (useLBRYUploader) {
|
2021-12-23 12:21:25 -06:00
|
|
|
publishPayload.tags.push(LBRY_FIRST_TAG);
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
}
|
|
|
|
|
2021-12-30 13:36:28 -06:00
|
|
|
// Set release time to the newly edited time.
|
|
|
|
// On edits, if not explicitly set to anytime, keep the original release/transaction time as release_time
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
if (releaseTimeEdited) {
|
|
|
|
publishPayload.release_time = releaseTimeEdited;
|
2021-12-30 13:36:28 -06:00
|
|
|
} else if (!releaseAnytime && myClaimForUriEditing && myClaimForUriEditing.value.release_time) {
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
publishPayload.release_time = Number(myClaimForUri.value.release_time);
|
2021-12-30 13:36:28 -06:00
|
|
|
} else if (!releaseAnytime && myClaimForUriEditing && myClaimForUriEditing.timestamp) {
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
publishPayload.release_time = Number(myClaimForUriEditing.timestamp);
|
|
|
|
}
|
|
|
|
|
2021-12-30 13:36:28 -06:00
|
|
|
// Remove internal scheduled tag if it exists.
|
|
|
|
publishPayload.tags = publishPayload.tags.filter((tag) => tag !== SCHEDULED_LIVESTREAM_TAG);
|
|
|
|
|
|
|
|
// Add internal scheduled tag if claim is a livestream and is being scheduled in the future.
|
2021-12-29 10:57:54 -06:00
|
|
|
if (isLivestreamPublish && publishPayload.release_time > nowTimeStamp) {
|
|
|
|
publishPayload.tags.push(SCHEDULED_LIVESTREAM_TAG);
|
|
|
|
}
|
|
|
|
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
if (channelId) {
|
|
|
|
publishPayload.channel_id = channelId;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (myClaimForUriEditing && myClaimForUriEditing.value && myClaimForUriEditing.value.locations) {
|
|
|
|
publishPayload.locations = myClaimForUriEditing.value.locations;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!contentIsFree && fee && fee.currency && Number(fee.amount) > 0) {
|
|
|
|
publishPayload.fee_currency = fee.currency;
|
|
|
|
publishPayload.fee_amount = creditsToString(fee.amount);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (optimize) {
|
|
|
|
publishPayload.optimize_file = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only pass file on new uploads, not metadata only edits.
|
|
|
|
// The sdk will figure it out
|
|
|
|
if (filePath && !isLivestreamPublish) {
|
|
|
|
publishPayload.file_path = filePath;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (preview) {
|
|
|
|
publishPayload.preview = true;
|
|
|
|
publishPayload.optimize_file = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return publishPayload;
|
|
|
|
}
|
|
|
|
|
2020-11-16 14:09:00 -05:00
|
|
|
export const doPublishDesktop = (filePath: string, preview?: boolean) => (dispatch: Dispatch, getState: () => {}) => {
|
2021-03-24 01:22:02 -04:00
|
|
|
const publishPreview = (previewResponse) => {
|
2020-07-31 21:33:49 +08:00
|
|
|
dispatch(
|
|
|
|
doOpenModal(MODALS.PUBLISH_PREVIEW, {
|
|
|
|
previewResponse,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2021-03-29 19:05:18 -04:00
|
|
|
const noFileParam = !filePath || filePath === NO_FILE;
|
|
|
|
const state = getState();
|
|
|
|
const editingUri = makeSelectPublishFormValue('editingURI')(state) || '';
|
2021-04-14 00:06:11 -04:00
|
|
|
const remoteUrl = makeSelectPublishFormValue('remoteFileUrl')(state);
|
2021-03-29 19:05:18 -04:00
|
|
|
const claim = makeSelectClaimForUri(editingUri)(state) || {};
|
|
|
|
const hasSourceFile = claim.value && claim.value.source;
|
2021-04-14 00:06:11 -04:00
|
|
|
const redirectToLivestream = noFileParam && !hasSourceFile && !remoteUrl;
|
2021-03-26 18:03:16 -04:00
|
|
|
|
2020-07-09 10:58:43 -04:00
|
|
|
const publishSuccess = (successResponse, lbryFirstError) => {
|
2019-07-12 10:58:24 -04:00
|
|
|
const state = getState();
|
2019-07-12 22:59:45 -04:00
|
|
|
const myClaims = selectMyClaims(state);
|
2019-04-24 10:02:08 -04:00
|
|
|
const pendingClaim = successResponse.outputs[0];
|
2019-10-16 17:36:50 -04:00
|
|
|
analytics.apiLogPublish(pendingClaim);
|
2019-10-11 21:55:54 -04:00
|
|
|
const { permanent_url: url } = pendingClaim;
|
2018-11-02 14:33:00 -04:00
|
|
|
const actions = [];
|
|
|
|
|
2019-11-13 10:59:34 -05:00
|
|
|
// @if TARGET='app'
|
2020-07-23 13:02:07 -04:00
|
|
|
actions.push(push(`/$/${PAGES.UPLOADS}`));
|
2019-11-13 10:59:34 -05:00
|
|
|
// @endif
|
2019-11-07 14:39:22 -05:00
|
|
|
|
2018-11-02 14:33:00 -04:00
|
|
|
actions.push({
|
2018-04-06 02:00:36 -04:00
|
|
|
type: ACTIONS.PUBLISH_SUCCESS,
|
2021-10-21 23:17:17 +08:00
|
|
|
data: {
|
|
|
|
type: resolveClaimTypeForAnalytics(pendingClaim),
|
|
|
|
},
|
2018-04-06 02:00:36 -04:00
|
|
|
});
|
2021-03-29 19:05:18 -04:00
|
|
|
|
2018-11-02 14:33:00 -04:00
|
|
|
// We have to fake a temp claim until the new pending one is returned by claim_list_mine
|
|
|
|
// We can't rely on claim_list_mine because there might be some delay before the new claims are returned
|
|
|
|
// Doing this allows us to show the pending claim immediately, it will get overwritten by the real one
|
2021-03-24 01:22:02 -04:00
|
|
|
const isMatch = (claim) => claim.claim_id === pendingClaim.claim_id;
|
2018-11-02 14:33:00 -04:00
|
|
|
const isEdit = myClaims.some(isMatch);
|
2019-06-24 23:05:46 -05:00
|
|
|
|
2020-04-24 09:51:00 -04:00
|
|
|
actions.push({
|
2021-10-17 16:36:14 +08:00
|
|
|
type: ACTIONS.UPDATE_PENDING_CLAIMS,
|
2020-04-24 09:51:00 -04:00
|
|
|
data: {
|
|
|
|
claims: [pendingClaim],
|
|
|
|
},
|
|
|
|
});
|
2020-05-07 08:22:55 -04:00
|
|
|
// @if TARGET='app'
|
|
|
|
actions.push({
|
2021-10-17 16:36:14 +08:00
|
|
|
type: ACTIONS.ADD_FILES_REFLECTING,
|
2020-05-07 08:22:55 -04:00
|
|
|
data: pendingClaim,
|
|
|
|
});
|
|
|
|
// @endif
|
2020-04-24 09:51:00 -04:00
|
|
|
|
|
|
|
dispatch(batchActions(...actions));
|
|
|
|
dispatch(
|
2019-11-13 10:59:34 -05:00
|
|
|
doOpenModal(MODALS.PUBLISH, {
|
|
|
|
uri: url,
|
|
|
|
isEdit,
|
|
|
|
filePath,
|
2020-07-09 10:58:43 -04:00
|
|
|
lbryFirstError,
|
2019-11-13 10:59:34 -05:00
|
|
|
})
|
|
|
|
);
|
2020-07-23 10:22:57 -04:00
|
|
|
dispatch(doCheckPendingClaims());
|
2020-05-07 08:22:55 -04:00
|
|
|
// @if TARGET='app'
|
|
|
|
dispatch(doCheckReflectingFiles());
|
|
|
|
// @endif
|
2021-03-29 19:05:18 -04:00
|
|
|
// @if TARGET='web'
|
|
|
|
if (redirectToLivestream) {
|
|
|
|
dispatch(push(`/$/${PAGES.LIVESTREAM}`));
|
|
|
|
}
|
|
|
|
// @endif
|
2018-04-06 02:00:36 -04:00
|
|
|
};
|
2018-03-26 14:32:43 -07:00
|
|
|
|
2021-03-24 01:22:02 -04:00
|
|
|
const publishFail = (error) => {
|
2019-07-12 10:58:24 -04:00
|
|
|
const actions = [];
|
2019-11-13 10:59:34 -05:00
|
|
|
actions.push({
|
|
|
|
type: ACTIONS.PUBLISH_FAIL,
|
|
|
|
});
|
2019-07-12 10:58:24 -04:00
|
|
|
actions.push(doError(error.message));
|
|
|
|
dispatch(batchActions(...actions));
|
2018-03-26 14:32:43 -07:00
|
|
|
};
|
2019-07-09 02:02:08 -04:00
|
|
|
|
2020-07-31 21:33:49 +08:00
|
|
|
if (preview) {
|
|
|
|
dispatch(doPublish(publishSuccess, publishFail, publishPreview));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-13 10:59:34 -05:00
|
|
|
// Redirect on web immediately because we have a file upload progress componenet
|
|
|
|
// on the publishes page. This doesn't exist on desktop so wait until we get a response
|
|
|
|
// from the SDK
|
|
|
|
// @if TARGET='web'
|
2021-03-29 19:05:18 -04:00
|
|
|
if (!redirectToLivestream) {
|
|
|
|
dispatch(push(`/$/${PAGES.UPLOADS}`));
|
|
|
|
}
|
2019-11-13 10:59:34 -05:00
|
|
|
// @endif
|
|
|
|
|
2019-10-23 15:39:51 -04:00
|
|
|
dispatch(doPublish(publishSuccess, publishFail));
|
2018-03-26 14:32:43 -07:00
|
|
|
};
|
2021-10-17 16:36:14 +08:00
|
|
|
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
export const doPublishResume = (publishPayload: any) => (dispatch: Dispatch, getState: () => {}) => {
|
|
|
|
const publishSuccess = (successResponse, lbryFirstError) => {
|
|
|
|
const state = getState();
|
|
|
|
const myClaimIds: Set<string> = selectMyActiveClaims(state);
|
|
|
|
|
|
|
|
const pendingClaim = successResponse.outputs[0];
|
|
|
|
const { permanent_url: url } = pendingClaim;
|
|
|
|
|
|
|
|
analytics.apiLogPublish(pendingClaim);
|
|
|
|
|
|
|
|
// We have to fake a temp claim until the new pending one is returned by claim_list_mine
|
|
|
|
// We can't rely on claim_list_mine because there might be some delay before the new claims are returned
|
|
|
|
// Doing this allows us to show the pending claim immediately, it will get overwritten by the real one
|
|
|
|
const isEdit = myClaimIds.has(pendingClaim.claim_id);
|
|
|
|
|
|
|
|
const actions = [];
|
|
|
|
|
|
|
|
actions.push({
|
|
|
|
type: ACTIONS.PUBLISH_SUCCESS,
|
|
|
|
data: {
|
|
|
|
type: resolveClaimTypeForAnalytics(pendingClaim),
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
actions.push({
|
|
|
|
type: ACTIONS.UPDATE_PENDING_CLAIMS,
|
|
|
|
data: {
|
|
|
|
claims: [pendingClaim],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
dispatch(batchActions(...actions));
|
|
|
|
|
|
|
|
dispatch(
|
|
|
|
doOpenModal(MODALS.PUBLISH, {
|
|
|
|
uri: url,
|
|
|
|
isEdit,
|
|
|
|
lbryFirstError,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
dispatch(doCheckPendingClaims());
|
|
|
|
};
|
|
|
|
|
|
|
|
const publishFail = (error) => {
|
|
|
|
const actions = [];
|
|
|
|
actions.push({
|
|
|
|
type: ACTIONS.PUBLISH_FAIL,
|
|
|
|
});
|
|
|
|
actions.push(doError(error.message));
|
|
|
|
dispatch(batchActions(...actions));
|
|
|
|
};
|
|
|
|
|
|
|
|
dispatch(doPublish(publishSuccess, publishFail, false, publishPayload));
|
|
|
|
};
|
|
|
|
|
2021-10-17 16:36:14 +08:00
|
|
|
export const doResetThumbnailStatus = () => (dispatch: Dispatch) => {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: {
|
|
|
|
thumbnailPath: '',
|
|
|
|
thumbnailError: undefined,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2021-12-16 16:16:22 +08:00
|
|
|
return fetch(SPEECH_STATUS)
|
2021-10-17 16:36:14 +08:00
|
|
|
.then((res) => res.json())
|
2021-12-16 16:16:22 +08:00
|
|
|
.then((status) => {
|
|
|
|
if (status.disabled) {
|
2021-10-17 16:36:14 +08:00
|
|
|
throw Error();
|
|
|
|
}
|
|
|
|
|
|
|
|
return dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: {
|
|
|
|
uploadThumbnailStatus: THUMBNAIL_STATUSES.READY,
|
|
|
|
thumbnail: '',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.catch(() =>
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: {
|
|
|
|
uploadThumbnailStatus: THUMBNAIL_STATUSES.API_DOWN,
|
|
|
|
thumbnail: '',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const doClearPublish = () => (dispatch: Dispatch) => {
|
|
|
|
dispatch({ type: ACTIONS.CLEAR_PUBLISH });
|
|
|
|
return dispatch(doResetThumbnailStatus());
|
|
|
|
};
|
|
|
|
|
|
|
|
export const doUpdatePublishForm = (publishFormValue: UpdatePublishFormData) => (dispatch: Dispatch) =>
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: { ...publishFormValue },
|
|
|
|
});
|
|
|
|
|
|
|
|
export const doUploadThumbnail = (
|
|
|
|
filePath?: string,
|
|
|
|
thumbnailBlob?: File,
|
|
|
|
fsAdapter?: any,
|
|
|
|
fs?: any,
|
|
|
|
path?: any,
|
|
|
|
cb?: (string) => void
|
|
|
|
) => (dispatch: Dispatch) => {
|
2021-12-16 16:16:22 +08:00
|
|
|
const downMessage = __('Thumbnail upload service may be down, try again later.');
|
2021-10-17 16:36:14 +08:00
|
|
|
let thumbnail, fileExt, fileName, fileType;
|
|
|
|
|
2021-12-16 16:16:22 +08:00
|
|
|
const makeid = () => {
|
|
|
|
let text = '';
|
|
|
|
const possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
|
|
|
for (let i = 0; i < 24; i += 1) text += possible.charAt(Math.floor(Math.random() * 62));
|
|
|
|
return text;
|
|
|
|
};
|
|
|
|
|
2021-10-17 16:36:14 +08:00
|
|
|
const uploadError = (error = '') => {
|
|
|
|
dispatch(
|
|
|
|
batchActions(
|
|
|
|
{
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: {
|
|
|
|
uploadThumbnailStatus: THUMBNAIL_STATUSES.READY,
|
|
|
|
thumbnail: '',
|
|
|
|
nsfw: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
doError(error)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
2021-12-16 16:16:22 +08:00
|
|
|
data: {
|
|
|
|
thumbnailError: undefined,
|
|
|
|
},
|
2021-10-17 16:36:14 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const doUpload = (data) => {
|
2021-12-16 16:16:22 +08:00
|
|
|
return fetch(SPEECH_PUBLISH, {
|
2021-10-17 16:36:14 +08:00
|
|
|
method: 'POST',
|
|
|
|
body: data,
|
|
|
|
})
|
|
|
|
.then((res) => res.text())
|
|
|
|
.then((text) => (text.length ? JSON.parse(text) : {}))
|
|
|
|
.then((json) => {
|
2021-12-16 16:16:22 +08:00
|
|
|
if (!json.success) return uploadError(json.message || downMessage);
|
2021-10-17 16:36:14 +08:00
|
|
|
if (cb) {
|
2021-12-16 16:16:22 +08:00
|
|
|
cb(json.data.serveUrl);
|
2021-10-17 16:36:14 +08:00
|
|
|
}
|
|
|
|
return dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: {
|
|
|
|
uploadThumbnailStatus: THUMBNAIL_STATUSES.COMPLETE,
|
2021-12-16 16:16:22 +08:00
|
|
|
thumbnail: json.data.serveUrl,
|
2021-10-17 16:36:14 +08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
let message = err.message;
|
|
|
|
|
|
|
|
// This sucks but ¯\_(ツ)_/¯
|
|
|
|
if (message === 'Failed to fetch') {
|
2021-12-16 16:16:22 +08:00
|
|
|
message = downMessage;
|
2021-10-17 16:36:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
uploadError(message);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_PUBLISH_FORM,
|
|
|
|
data: { uploadThumbnailStatus: THUMBNAIL_STATUSES.IN_PROGRESS },
|
|
|
|
});
|
|
|
|
|
|
|
|
if (fsAdapter && fsAdapter.readFile && filePath) {
|
|
|
|
fsAdapter.readFile(filePath, 'base64').then((base64Image) => {
|
|
|
|
fileExt = 'png';
|
|
|
|
fileName = 'thumbnail.png';
|
|
|
|
fileType = 'image/png';
|
|
|
|
|
|
|
|
const data = new FormData();
|
2021-12-16 16:16:22 +08:00
|
|
|
const name = makeid();
|
|
|
|
data.append('name', name);
|
2021-10-17 16:36:14 +08:00
|
|
|
// $FlowFixMe
|
2021-12-16 16:16:22 +08:00
|
|
|
data.append('file', { uri: 'file://' + filePath, type: fileType, name: fileName });
|
2021-10-17 16:36:14 +08:00
|
|
|
return doUpload(data);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
if (filePath && fs && path) {
|
|
|
|
thumbnail = fs.readFileSync(filePath);
|
|
|
|
fileExt = path.extname(filePath);
|
|
|
|
fileName = path.basename(filePath);
|
|
|
|
fileType = `image/${fileExt.slice(1)}`;
|
|
|
|
} else if (thumbnailBlob) {
|
|
|
|
fileExt = `.${thumbnailBlob.type && thumbnailBlob.type.split('/')[1]}`;
|
|
|
|
fileName = thumbnailBlob.name;
|
|
|
|
fileType = thumbnailBlob.type;
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const data = new FormData();
|
2021-12-16 16:16:22 +08:00
|
|
|
const name = makeid();
|
2021-10-17 16:36:14 +08:00
|
|
|
const file = thumbnailBlob || (thumbnail && new File([thumbnail], fileName, { type: fileType }));
|
2021-12-16 16:16:22 +08:00
|
|
|
data.append('name', name);
|
2021-10-17 16:36:14 +08:00
|
|
|
// $FlowFixMe
|
2021-12-16 16:16:22 +08:00
|
|
|
data.append('file', file);
|
2021-10-17 16:36:14 +08:00
|
|
|
return doUpload(data);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const doPrepareEdit = (claim: StreamClaim, uri: string, fileInfo: FileListItem, fs: any) => (
|
|
|
|
dispatch: Dispatch
|
|
|
|
) => {
|
|
|
|
const { name, amount, value = {} } = claim;
|
|
|
|
const channelName = (claim && claim.signing_channel && claim.signing_channel.name) || null;
|
|
|
|
const {
|
|
|
|
author,
|
|
|
|
description,
|
|
|
|
// use same values as default state
|
|
|
|
// fee will be undefined for free content
|
|
|
|
fee = {
|
|
|
|
amount: '0',
|
|
|
|
currency: 'LBC',
|
|
|
|
},
|
|
|
|
languages,
|
|
|
|
release_time,
|
|
|
|
license,
|
|
|
|
license_url: licenseUrl,
|
|
|
|
thumbnail,
|
|
|
|
title,
|
|
|
|
tags,
|
|
|
|
} = value;
|
|
|
|
|
|
|
|
const publishData: UpdatePublishFormData = {
|
|
|
|
name,
|
|
|
|
bid: Number(amount),
|
|
|
|
contentIsFree: fee.amount === '0',
|
|
|
|
author,
|
|
|
|
description,
|
|
|
|
fee,
|
|
|
|
languages,
|
|
|
|
releaseTime: release_time,
|
|
|
|
releaseTimeEdited: undefined,
|
|
|
|
thumbnail: thumbnail ? thumbnail.url : null,
|
|
|
|
title,
|
|
|
|
uri,
|
|
|
|
uploadThumbnailStatus: thumbnail ? THUMBNAIL_STATUSES.MANUAL : undefined,
|
|
|
|
licenseUrl,
|
|
|
|
nsfw: isClaimNsfw(claim),
|
|
|
|
tags: tags ? tags.map((tag) => ({ name: tag })) : [],
|
|
|
|
};
|
|
|
|
|
|
|
|
// Make sure custom licenses are mapped properly
|
|
|
|
// If the license isn't one of the standard licenses, map the custom license and description/url
|
|
|
|
if (!CC_LICENSES.some(({ value }) => value === license)) {
|
|
|
|
if (!license || license === NONE || license === PUBLIC_DOMAIN) {
|
|
|
|
publishData.licenseType = license;
|
|
|
|
} else if (license && !licenseUrl && license !== NONE) {
|
|
|
|
publishData.licenseType = COPYRIGHT;
|
|
|
|
} else {
|
|
|
|
publishData.licenseType = OTHER;
|
|
|
|
}
|
|
|
|
|
|
|
|
publishData.otherLicenseDescription = license;
|
|
|
|
} else {
|
|
|
|
publishData.licenseType = license;
|
|
|
|
}
|
|
|
|
if (channelName) {
|
|
|
|
publishData['channel'] = channelName;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatch({ type: ACTIONS.DO_PREPARE_EDIT, data: publishData });
|
|
|
|
};
|
|
|
|
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
export const doPublish = (success: Function, fail: Function, preview: Function, payload: any) => (
|
2021-10-17 16:36:14 +08:00
|
|
|
dispatch: Dispatch,
|
|
|
|
getState: () => {}
|
|
|
|
) => {
|
|
|
|
if (!preview) {
|
|
|
|
dispatch({ type: ACTIONS.PUBLISH_START });
|
|
|
|
}
|
|
|
|
|
|
|
|
const state = getState();
|
|
|
|
const myClaimForUri = selectMyClaimForUri(state);
|
|
|
|
const myChannels = selectMyChannelClaims(state);
|
|
|
|
// const myClaims = selectMyClaimsWithoutChannels(state);
|
|
|
|
// get redux publish form
|
|
|
|
const publishData = selectPublishFormValues(state);
|
|
|
|
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
const publishPayload = payload || resolvePublishPayload(publishData, myClaimForUri, myChannels, preview);
|
2021-10-17 16:36:14 +08:00
|
|
|
|
|
|
|
if (preview) {
|
|
|
|
return Lbry.publish(publishPayload).then((previewResponse: PublishResponse) => {
|
|
|
|
return preview(previewResponse);
|
|
|
|
}, fail);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Lbry.publish(publishPayload).then((response: PublishResponse) => {
|
|
|
|
// TODO: Restore LbryFirst
|
|
|
|
// if (!useLBRYUploader) {
|
|
|
|
return success(response);
|
|
|
|
// }
|
|
|
|
|
|
|
|
// $FlowFixMe
|
|
|
|
// publishPayload.permanent_url = response.outputs[0].permanent_url;
|
|
|
|
//
|
|
|
|
// return LbryFirst.upload(publishPayload)
|
|
|
|
// .then(() => {
|
|
|
|
// // Return original publish response so app treats it like a normal publish
|
|
|
|
// return success(response);
|
|
|
|
// })
|
|
|
|
// .catch((error) => {
|
|
|
|
// return success(response, error);
|
|
|
|
// });
|
|
|
|
}, fail);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Calls file_list until any reflecting files are done
|
|
|
|
export const doCheckReflectingFiles = () => (dispatch: Dispatch, getState: GetState) => {
|
|
|
|
const state = getState();
|
|
|
|
const { checkingReflector } = state.claims;
|
|
|
|
let reflectorCheckInterval;
|
|
|
|
|
|
|
|
const checkFileList = async () => {
|
|
|
|
const state = getState();
|
|
|
|
const reflectingById = selectReflectingById(state);
|
|
|
|
const ids = Object.keys(reflectingById);
|
|
|
|
|
|
|
|
const newReflectingById = {};
|
|
|
|
const promises = [];
|
|
|
|
// TODO: just use file_list({claim_id: Array<claimId>})
|
|
|
|
if (Object.keys(reflectingById).length) {
|
|
|
|
ids.forEach((claimId) => {
|
|
|
|
promises.push(Lbry.file_list({ claim_id: claimId }));
|
|
|
|
});
|
|
|
|
|
|
|
|
Promise.all(promises)
|
|
|
|
.then((results) => {
|
|
|
|
results.forEach((res) => {
|
|
|
|
if (res.items[0]) {
|
|
|
|
const fileListItem = res.items[0];
|
|
|
|
const fileClaimId = fileListItem.claim_id;
|
|
|
|
const {
|
|
|
|
is_fully_reflected: done,
|
|
|
|
uploading_to_reflector: uploading,
|
|
|
|
reflector_progress: progress,
|
|
|
|
} = fileListItem;
|
|
|
|
if (uploading) {
|
|
|
|
newReflectingById[fileClaimId] = {
|
|
|
|
fileListItem: fileListItem,
|
|
|
|
progress,
|
|
|
|
stalled: !done && !uploading,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_FILES_REFLECTING,
|
|
|
|
data: newReflectingById,
|
|
|
|
});
|
|
|
|
if (!Object.keys(newReflectingById).length) {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.TOGGLE_CHECKING_REFLECTING,
|
|
|
|
data: false,
|
|
|
|
});
|
|
|
|
clearInterval(reflectorCheckInterval);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.TOGGLE_CHECKING_REFLECTING,
|
|
|
|
data: false,
|
|
|
|
});
|
|
|
|
clearInterval(reflectorCheckInterval);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
// do it once...
|
|
|
|
checkFileList();
|
|
|
|
// then start the interval if it's not already started
|
|
|
|
if (!checkingReflector) {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.TOGGLE_CHECKING_REFLECTING,
|
|
|
|
data: true,
|
|
|
|
});
|
|
|
|
reflectorCheckInterval = setInterval(() => {
|
|
|
|
checkFileList();
|
|
|
|
}, 5000);
|
|
|
|
}
|
|
|
|
};
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
|
|
|
|
export function doUpdateUploadAdd(
|
|
|
|
file: File | string,
|
|
|
|
params: { [key: string]: any },
|
|
|
|
uploader: TusUploader | XMLHttpRequest
|
|
|
|
) {
|
|
|
|
return (dispatch: Dispatch, getState: GetState) => {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_UPLOAD_ADD,
|
|
|
|
data: { file, params, uploader },
|
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-12-07 06:48:09 -08:00
|
|
|
export const doUpdateUploadProgress = (props: { guid: string, progress?: string, status?: string }) => (
|
|
|
|
dispatch: Dispatch
|
|
|
|
) =>
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_UPLOAD_PROGRESS,
|
|
|
|
data: props,
|
|
|
|
});
|
|
|
|
|
2021-12-07 06:48:09 -08:00
|
|
|
/**
|
|
|
|
* doUpdateUploadRemove
|
|
|
|
*
|
|
|
|
* @param guid
|
|
|
|
* @param params Optional. Retain to allow removal of old keys, which are
|
|
|
|
* derived from `name#channel` instead of using a guid.
|
|
|
|
* Can be removed after January 2022.
|
|
|
|
* @returns {(function(Dispatch, GetState): void)|*}
|
|
|
|
*/
|
|
|
|
export function doUpdateUploadRemove(guid: string, params?: { [key: string]: any }) {
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
return (dispatch: Dispatch, getState: GetState) => {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_UPLOAD_REMOVE,
|
2021-12-07 06:48:09 -08:00
|
|
|
data: { guid, params },
|
Support resume-able upload via tus (#186)
* Publish button: use spinner instead of "Publishing..."
Looks better, plus the preview could take a while sometimes.
* Refactor `doPublish`. No functional change
This is to allow `doPublish` to accept a custom payload as an input (for resuming uploads), instead of always resolving it from the redux data.
* Add doPublishResume
* Support resume-able upload via tus
## Issue
38 Handle resumable file upload
## Notes
Since we can't serialize a File object, we'll need to the user to re-select the file to resume.
* Exclude "modified date" for Firefox/Android
## Issue
It appears that the modification date of the Android file changes when selected, so that file was deemed "different" when trying to resume upload.
## Change
Exclude modification date for now. Let's assume a smart user.
* Move 'currentUploads' to 'publish' reducer
`publish` is currently rehydrated, so we can ride on that and don't need to store the `currentUploads` in `localStorage` for persistence. This would allow us to store Markdown Post data too, as `localStorage` has a 5MB limit per app.
We could have also made `webReducer` rehydrate, but in this repo, there is no need to split it to another reducer. It also makes more sense to be part of publish anyway (at least to me).
This change is mostly moving items between files, with the exception of
1. An additional REHYDRATE in the publish reducer to clean up the tusUploader.
2. Not clearing `currentUploads` in CLEAR_PUBLISH.
* Restore v1 code for livestream replay, etc.
v2 (tus) does not handle `remote_url`, so the app still needs v1 for that. Since we'll still have v1 code, use v1 for previews as well.
2021-11-11 02:16:16 +08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|