-
Notifications
You must be signed in to change notification settings - Fork 118
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Memory leak when we interrupt the PDF loading #160
Comments
Hi @xandao-dev, Thanks for discovering this issue. I will try to fix it, but I would also appreciate the help in resolving it. |
@hrynko hey! I was able to reproduce the issue in your project by slightly modifying the demo app to: <template>
<div>
<button @click="enabled = !enabled">
{{ enabled ? 'Disable' : 'Enable' }}
</button>
<vue-pdf-embed
v-if="enabled"
:image-resources-path="annotationIconsPath"
:source="pdfSource"
/>
</div>
</template>
<script>
import VuePdfEmbed from '../src/vue-pdf-embed.vue'
export default {
components: {
VuePdfEmbed,
},
data() {
return {
enabled: true,
annotationIconsPath: '/node_modules/pdfjs-dist/web/images/',
pdfSource: '/demo/sample-50-MB-pdf-file.pdf',
}
},
}
</script> Note the pdf source of 50MB, we have to add it to the demo folder. Here is the link for the one I used: https://www.learningcontainer.com/sample-pdf-files-for-testing/ To check the memory leak, just spam clicks on the Enable/Disable button, and watch the detached elements tool. |
Looking closer at the first image I sent, it seems to be a problem with the pdf.GlobalWorkerOptions.workerPort By moving the creation of the pdf worker instance to the Vue created lifecycle and then correctly terminating the worker in the beforeDestroy lifecycle hook, we can ensure proper worker termination. I also read this issue about properly closing the loading task, but I haven't tested: mozilla/pdf.js#11595 |
@hrynko Can you please check if there are any drawbacks to loading the instance during the created lifecycle? It seems to be the right way for me. |
Hi @xandao-dev, Thank you for opening the PR, it looks good to me. I don't see any drawbacks in moving the worker initialization, however could you comment on the motivation for doing so? |
Sure, the worker does not get initialized again after termination in the beforeDestroy/beforeUnmount hook. Therefore, when the component is loaded again, it needs to be started again. And most important, it avoids terminating a worker from another component. An important detail is that before we only had 1 instance, moving into the vue lifecycle we started to create one instance per component. |
If the worker is heavy, maybe it's an overkill to have an instance for each component. We do have alternatives, for example: Stick with only one "global" (not really global, it belongs to a closure) worker, and in the beforeDestroy hook, we destroy the loading tasks instead of the worker. Like so: loadingTask.destroy(); The option I followed is more memory-safe I guess, but it's also a bit more heavy (I don't know how much heavier). |
@hrynko hold on a bit, I would like to investigate a bit more with multiple components. |
The workers are not being removed (in that first solution should exist one per pdf): ![]() I was dumb, GlobalWorkerOption has Global in its name for a reason. If you would like to use a worker for each instance you have to update other parts of the code. I tried to experiment with it to see if it handles better parallel PDF loading, but I wasn't able to accomplish it. So let's take a step back and analyze the memory leaks again: The memory leaks were caused by event listeners, by cleaning them it was fixed. I used this.documentLoadingTask.onProgress instead of this.document.loadingTask.onProgress because the latter was sometimes unreachable, possibly due to the promise not being fulfilled yet. |
Here is the template I used to test the memory leak and the issues with multiple pdfs: <template>
<div class="container">
<div class="minw">
<button @click="enabled = !enabled">
{{ enabled ? 'Disable 1' : 'Enable 1' }}
</button>
<vue-pdf-embed
v-if="enabled"
:image-resources-path="annotationIconsPath"
:source="pdfSource"
/>
</div>
<div class="minw">
<button @click="enabled2 = !enabled2">
{{ enabled2 ? 'Disable 2' : 'Enable 2' }}
</button>
<vue-pdf-embed
v-if="enabled2"
:image-resources-path="annotationIconsPath"
:source="pdfSource"
/>
</div>
<div class="minw">
<button @click="enabled3 = !enabled3">
{{ enabled3 ? 'Disable 3' : 'Enable 3' }}
</button>
<vue-pdf-embed
v-if="enabled3"
:image-resources-path="annotationIconsPath"
:source="pdfSource"
/>
</div>
<button
@click="
enabled3 = !enabled3
enabled2 = !enabled2
enabled = !enabled
"
>
Toggle All
</button>
</div>
</template>
<script>
import VuePdfEmbed from '../src/vue-pdf-embed.vue'
export default {
components: {
VuePdfEmbed,
},
data() {
return {
enabled: true,
enabled2: true,
enabled3: true,
annotationIconsPath: '/node_modules/pdfjs-dist/web/images/',
pdfSource: '/demo/sample-50-MB-pdf-file.pdf',
}
},
}
</script>
<style lang="scss">
body {
padding: 16px;
background-color: #ccc;
}
.container {
display: flex;
justify-content: center;
gap: 16px;
max-width: 480px;
margin: auto;
}
.minw {
min-width: 300px;
}
.vue-pdf-embed {
margin: auto;
max-width: 480px;
& > div {
margin-bottom: 4px;
box-shadow: 0 2px 8px 4px rgba(0, 0, 0, 0.1);
}
}
</style> |
So does this fix the initial issue? If so, could you please update the PR? |
@hrynko all done, PR updated. |
The issue seems to be back with your latest commit (still throwing |
@hrynko It's fine here. Should I've seen anything in the console? ![]() ![]() ps. This detached element is from an extension and is not a memory leak. |
Hmmm, can you share the pdf, please? This error is not happening here. |
I applied the latest changes from the PR and the demo code as above. The PDF used can be found by the following link: https://history.nasa.gov/alsj/a17/A17_FlightPlan.pdf The only thing I added is the logging of the |
Ok, it's clear now. Is there anything else you would like to change/add in this PR? |
@hrynko we are good to go, thanks. |
@hrynko Hey, don't want to bother you, but when can I expect a release? |
Hi @xandao-dev, It has already been released in |
How to reproduce
Observations
Tools used to identify the Memory Leak: Edge Detached Elements.
Our application: A SPA and I this specific page we change the path by calling $router.push.
Our implementation:
data:image/s3,"s3://crabby-images/f8339/f8339585518882f3a4f90584db76a1aaac0f8c34" alt="image"
The text was updated successfully, but these errors were encountered: