stable main" >> /etc/apt/sources.list.d/google.list' \\
&& apt-get update \\
&& apt-get install google-chrome-stable -y --no-install-recommends \\
&& rm -rf /var/lib/apt/lists/*
Additionally, Puppeteer needed specific launch arguments to work in a containerized environment:
1browserInstance = await puppeteer.launch({
2 executablePath,
3 args: [
4 "--no-sandbox",
5 "--disable-setuid-sandbox",
6 "--disable-dev-shm-usage",
7 // Other arguments...
8 ],
9 headless: true,
10});
Rather than using service account keys (which can be a security risk), I implemented Workload Identity Federation for secure authentication between GitHub Actions and Google Cloud:
1- id: 'auth'
2 name: 'Authenticate to Google Cloud'
3 uses: 'google-github-actions/auth@v2'
4 with:
5 workload_identity_provider: 'projects/PROJECT_NUMBER/locations/global/workloadIdentityPools/github-pool/providers/github-provider'
6 service_account: 'github-actions-sa@cn-math-worksheet-generator.iam.gserviceaccount.com'
This approach eliminated the need to store service account keys as GitHub secrets, significantly improving security.
PDF generation is resource-intensive, especially when creating multiple worksheets. I had to carefully tune the Cloud Run resource allocation to ensure reliable performance:
1resources:
2 limits:
3 memory: 1Gi
4 cpu: 1000m
Through testing, I found that 1GB of memory and 1 CPU core provided the optimal balance between cost and performance for this application.
I designed the application with progressive enhancement in mind, ensuring it works even if JavaScript is disabled or fails to load:
The application needed to work well on various devices, from desktop computers to tablets and phones:
1@media (max-width: 768px) {
2 .app-main {
3 flex-direction: column;
4 }
56 .settings-panel, .preview-panel {
7 width: 100%;
8 }
9}
This responsive approach ensures teachers can create worksheets from any device, whether they're at their desk or moving around the classroom.
Accessibility was a priority from the beginning:
1<div
2 class="checkbox-group"
3 role="group"
4 aria-labelledby="operations-heading"
5>
6 <label class="checkbox-container">
7 <input type="checkbox" id="addition" checked>
8 <span class="checkbox-label">Addition</span>
9 </label>
10 <!-- Other checkboxes... -->
11</div>
I used semantic HTML, ARIA attributes, and proper focus management to ensure the application is usable by everyone, including those using screen readers or keyboard navigation.
Before writing any code, I spent time understanding exactly what teachers and parents needed in a worksheet generator. This user-focused approach helped me prioritize features and create an intuitive interface.
While I initially considered using a full-stack framework like Next.js, I ultimately chose a simpler approach with Deno and vanilla JavaScript. This decision paid off in terms of performance and deployment simplicity.
I encountered several issues with Puppeteer in Docker that weren't apparent in local development. Setting up a production-like environment early in the development process would have saved time troubleshooting these issues.
Implementing Workload Identity Federation was more complex than using service account keys, but the security benefits were worth the effort. Building security into the application from the beginning is always easier than adding it later.
The shared browser instance pattern was a significant performance improvement, but it came after initial deployment. Continuous monitoring and optimization are essential for maintaining a fast, responsive application.
While the current version of the Math Worksheet Generator meets the initial requirements, there are several enhancements I'm considering for future updates:
Building the Math Worksheet Generator was a rewarding experience that combined my passion for education with my love of software development. The application has already helped numerous teachers save time creating math practice materials, and I'm excited to continue improving it based on user feedback.
The project demonstrates that even relatively simple applications can benefit from modern development practices like containerization, CI/CD automation, and serverless deployment. By focusing on user needs and leveraging the right technologies, it's possible to create valuable tools that make a real difference in people's daily work.
If you're interested in exploring the code or contributing to the project, check out the GitHub repository or try the live application.
Happy teaching and coding!
As always, thank you for reading. I really appreciate it. 💖