1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-11 18:24:43 +02:00

All: Handle case where resource blob is missing during sync

This commit is contained in:
Laurent Cozic 2018-01-21 17:01:37 +00:00
parent e355f4e49b
commit 86eee376bb
9 changed files with 47 additions and 19 deletions

View File

@ -683,12 +683,12 @@ describe('Synchronizer', function() {
await switchClient(2); await switchClient(2);
synchronizer().debugFlags_ = ['cancelDeltaLoop2']; synchronizer().testingHooks_ = ['cancelDeltaLoop2'];
let context = await synchronizer().start(); let context = await synchronizer().start();
let notes = await Note.all(); let notes = await Note.all();
expect(notes.length).toBe(0); expect(notes.length).toBe(0);
synchronizer().debugFlags_ = []; synchronizer().testingHooks_ = [];
await synchronizer().start({ context: context }); await synchronizer().start({ context: context });
notes = await Note.all(); notes = await Note.all();
expect(notes.length).toBe(1); expect(notes.length).toBe(1);
@ -702,9 +702,9 @@ describe('Synchronizer', function() {
let disabledItems = await BaseItem.syncDisabledItems(syncTargetId()); let disabledItems = await BaseItem.syncDisabledItems(syncTargetId());
expect(disabledItems.length).toBe(0); expect(disabledItems.length).toBe(0);
await Note.save({ id: noteId, title: "un mod", }); await Note.save({ id: noteId, title: "un mod", });
synchronizer().debugFlags_ = ['rejectedByTarget']; synchronizer().testingHooks_ = ['rejectedByTarget'];
await synchronizer().start(); await synchronizer().start();
synchronizer().debugFlags_ = []; synchronizer().testingHooks_ = [];
await synchronizer().start(); // Another sync to check that this item is now excluded from sync await synchronizer().start(); // Another sync to check that this item is now excluded from sync
await switchClient(2); await switchClient(2);

View File

@ -121,7 +121,7 @@ class EncryptionConfigScreenComponent extends React.Component {
} }
} }
const decryptedItemsInfo = this.props.encryptionEnabled ? <p style={theme.textStyle}>{shared.decryptedStatText(this)}</p> : null; const decryptedItemsInfo = <p style={theme.textStyle}>{shared.decryptedStatText(this)}</p>;
const toggleButton = <button onClick={() => { onToggleButtonClick() }}>{this.props.encryptionEnabled ? _('Disable encryption') : _('Enable encryption')}</button> const toggleButton = <button onClick={() => { onToggleButtonClick() }}>{this.props.encryptionEnabled ? _('Disable encryption') : _('Enable encryption')}</button>
let masterKeySection = null; let masterKeySection = null;

View File

@ -3,6 +3,15 @@
// Make it possible to require("/lib/...") without specifying full path // Make it possible to require("/lib/...") without specifying full path
require('app-module-path').addPath(__dirname); require('app-module-path').addPath(__dirname);
// Disable React message in console "Download the React DevTools for a better development experience"
// https://stackoverflow.com/questions/42196819/disable-hide-download-the-react-devtools#42196820
__REACT_DEVTOOLS_GLOBAL_HOOK__ = {
supportsFiber: true,
inject: function() {},
onCommitFiberRoot: function() {},
onCommitFiberUnmount: function() {},
};
const { app } = require('./app.js'); const { app } = require('./app.js');
const Folder = require('lib/models/Folder.js'); const Folder = require('lib/models/Folder.js');
const Resource = require('lib/models/Resource.js'); const Resource = require('lib/models/Resource.js');

View File

@ -73,7 +73,7 @@ class MdToHtml {
renderImage_(attrs, options) { renderImage_(attrs, options) {
const loadResource = async (id) => { const loadResource = async (id) => {
console.info('Loading resource: ' + id); // console.info('Loading resource: ' + id);
// Initially set to to an empty object to make // Initially set to to an empty object to make
// it clear that it is being loaded. Otherwise // it clear that it is being loaded. Otherwise

View File

@ -1,5 +1,7 @@
const { isHidden } = require('lib/path-utils.js'); const { isHidden } = require('lib/path-utils.js');
const { Logger } = require('lib/logger.js'); const { Logger } = require('lib/logger.js');
const { shim } = require('lib/shim');
const JoplinError = require('lib/JoplinError');
class FileApi { class FileApi {
@ -10,6 +12,10 @@ class FileApi {
this.syncTargetId_ = null; this.syncTargetId_ = null;
} }
fsDriver() {
return shim.fsDriver();
}
driver() { driver() {
return this.driver_; return this.driver_;
} }
@ -83,8 +89,13 @@ class FileApi {
return this.driver_.get(this.fullPath_(path), options); return this.driver_.get(this.fullPath_(path), options);
} }
put(path, content, options = null) { async put(path, content, options = null) {
this.logger().debug('put ' + this.fullPath_(path)); this.logger().debug('put ' + this.fullPath_(path), options);
if (options && options.source === 'file') {
if (!await this.fsDriver().exists(options.path)) throw new JoplinError('File not found: ' + options.path, 'fileNotFound');
}
return this.driver_.put(this.fullPath_(path), content, options); return this.driver_.put(this.fullPath_(path), content, options);
} }

View File

@ -4,14 +4,20 @@ const { GeolocationNode } = require('lib/geolocation-node.js');
const { FileApiDriverLocal } = require('lib/file-api-driver-local.js'); const { FileApiDriverLocal } = require('lib/file-api-driver-local.js');
const { time } = require('lib/time-utils.js'); const { time } = require('lib/time-utils.js');
const { setLocale, defaultLocale, closestSupportedLocale } = require('lib/locale.js'); const { setLocale, defaultLocale, closestSupportedLocale } = require('lib/locale.js');
const { FsDriverNode } = require('lib/fs-driver-node.js');
function shimInit() { function shimInit() {
shim.fs = fs; shim.fsDriver = () => { throw new Error('Not implemented') }
shim.FileApiDriverLocal = FileApiDriverLocal; shim.FileApiDriverLocal = FileApiDriverLocal;
shim.Geolocation = GeolocationNode; shim.Geolocation = GeolocationNode;
shim.FormData = require('form-data'); shim.FormData = require('form-data');
shim.sjclModule = require('lib/vendor/sjcl.js'); shim.sjclModule = require('lib/vendor/sjcl.js');
shim.fsDriver = () => {
if (!shim.fsDriver_) shim.fsDriver_ = new FsDriverNode();
return shim.fsDriver_;
}
shim.randomBytes = async (count) => { shim.randomBytes = async (count) => {
const buffer = require('crypto').randomBytes(count); const buffer = require('crypto').randomBytes(count);
return Array.from(buffer); return Array.from(buffer);

View File

@ -3,6 +3,7 @@ const { GeolocationReact } = require('lib/geolocation-react.js');
const { PoorManIntervals } = require('lib/poor-man-intervals.js'); const { PoorManIntervals } = require('lib/poor-man-intervals.js');
const RNFetchBlob = require('react-native-fetch-blob').default; const RNFetchBlob = require('react-native-fetch-blob').default;
const { generateSecureRandom } = require('react-native-securerandom'); const { generateSecureRandom } = require('react-native-securerandom');
const FsDriverRN = require('lib/fs-driver-rn.js').FsDriverRN;
function shimInit() { function shimInit() {
shim.Geolocation = GeolocationReact; shim.Geolocation = GeolocationReact;
@ -10,6 +11,11 @@ function shimInit() {
shim.clearInterval = PoorManIntervals.clearInterval; shim.clearInterval = PoorManIntervals.clearInterval;
shim.sjclModule = require('lib/vendor/sjcl-rn.js'); shim.sjclModule = require('lib/vendor/sjcl-rn.js');
shim.fsDriver = () => {
if (!shim.fsDriver_) shim.fsDriver_ = new FsDriverRN();
return shim.fsDriver_;
}
shim.randomBytes = async (count) => { shim.randomBytes = async (count) => {
const randomBytes = await generateSecureRandom(count); const randomBytes = await generateSecureRandom(count);
let temp = []; let temp = [];

View File

@ -107,7 +107,7 @@ shim.fetchWithRetry = async function(fetchFn, options = null) {
shim.nativeFetch_ = typeof fetch !== 'undefined' ? fetch : null; shim.nativeFetch_ = typeof fetch !== 'undefined' ? fetch : null;
shim.fetch = () => { throw new Error('Not implemented'); } shim.fetch = () => { throw new Error('Not implemented'); }
shim.FormData = typeof FormData !== 'undefined' ? FormData : null; shim.FormData = typeof FormData !== 'undefined' ? FormData : null;
shim.fs = null; shim.fsDriver = () => { throw new Error('Not implemented') }
shim.FileApiDriverLocal = null; shim.FileApiDriverLocal = null;
shim.readLocalFileBase64 = (path) => { throw new Error('Not implemented'); } shim.readLocalFileBase64 = (path) => { throw new Error('Not implemented'); }
shim.uploadBlob = () => { throw new Error('Not implemented'); } shim.uploadBlob = () => { throw new Error('Not implemented'); }

View File

@ -10,7 +10,7 @@ const { time } = require('lib/time-utils.js');
const { Logger } = require('lib/logger.js'); const { Logger } = require('lib/logger.js');
const { _ } = require('lib/locale.js'); const { _ } = require('lib/locale.js');
const { shim } = require('lib/shim.js'); const { shim } = require('lib/shim.js');
const moment = require('moment'); const JoplinError = require('lib/JoplinError');
class Synchronizer { class Synchronizer {
@ -27,7 +27,7 @@ class Synchronizer {
// Debug flags are used to test certain hard-to-test conditions // Debug flags are used to test certain hard-to-test conditions
// such as cancelling in the middle of a loop. // such as cancelling in the middle of a loop.
this.debugFlags_ = []; this.testingHooks_ = [];
this.onProgress_ = function(s) {}; this.onProgress_ = function(s) {};
this.progressReport_ = {}; this.progressReport_ = {};
@ -279,7 +279,7 @@ class Synchronizer {
const localResourceContentPath = result.path; const localResourceContentPath = result.path;
await this.api().put(remoteContentPath, null, { path: localResourceContentPath, source: 'file' }); await this.api().put(remoteContentPath, null, { path: localResourceContentPath, source: 'file' });
} catch (error) { } catch (error) {
if (error && error.code === 'rejectedByTarget') { if (error && ['rejectedByTarget', 'fileNotFound'].indexOf(error.code) >= 0) {
await handleCannotSyncItem(syncTargetId, local, error.message); await handleCannotSyncItem(syncTargetId, local, error.message);
action = null; action = null;
} else { } else {
@ -303,11 +303,7 @@ class Synchronizer {
let canSync = true; let canSync = true;
try { try {
if (this.debugFlags_.indexOf('rejectedByTarget') >= 0) { if (this.testingHooks_.indexOf('rejectedByTarget') >= 0) throw new JoplinError('Testing rejectedByTarget', 'rejectedByTarget');
const error = new Error('Testing rejectedByTarget');
error.code = 'rejectedByTarget';
throw error;
}
const content = await ItemClass.serializeForSync(local); const content = await ItemClass.serializeForSync(local);
await this.api().put(path, content); await this.api().put(path, content);
} catch (error) { } catch (error) {
@ -449,7 +445,7 @@ class Synchronizer {
this.logSyncOperation('fetchingTotal', null, null, 'Fetching delta items from sync target', remotes.length); this.logSyncOperation('fetchingTotal', null, null, 'Fetching delta items from sync target', remotes.length);
for (let i = 0; i < remotes.length; i++) { for (let i = 0; i < remotes.length; i++) {
if (this.cancelling() || this.debugFlags_.indexOf('cancelDeltaLoop2') >= 0) { if (this.cancelling() || this.testingHooks_.indexOf('cancelDeltaLoop2') >= 0) {
hasCancelled = true; hasCancelled = true;
break; break;
} }