mirror of
https://github.com/gotenberg/gotenberg.git
synced 2026-07-02 08:27:41 +08:00
b418f1eb05
* initial changes
* Add tests
* Fix edge case when we need to regenerate appearances
* Fix comments
* Add missing comment
* Add missing comment
* Add missing comment
* Add flatten option to the merge route
* Add flatten option to the libreoffice convert route
* Add flatten option to the chromium convert route
* Revert "Add flatten option to the chromium convert route"
This reverts commit cdab8b4e6b.
* Ignore lint false positives
* Add missing tests
* Add flatten route tests
* Replace input instead of creating a new file
* create copy before flatten in tests
---------
Co-authored-by: Peter Chakalov <peter.chakalov@abraxa.com>
137 lines
4.1 KiB
Go
137 lines
4.1 KiB
Go
package pdfcpu
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/gotenberg/gotenberg/v8/pkg/gotenberg"
|
|
)
|
|
|
|
func init() {
|
|
gotenberg.MustRegisterModule(new(PdfCpu))
|
|
}
|
|
|
|
// PdfCpu abstracts the CLI tool pdfcpu and implements the
|
|
// [gotenberg.PdfEngine] interface.
|
|
type PdfCpu struct {
|
|
binPath string
|
|
}
|
|
|
|
// Descriptor returns a [PdfCpu]'s module descriptor.
|
|
func (engine *PdfCpu) Descriptor() gotenberg.ModuleDescriptor {
|
|
return gotenberg.ModuleDescriptor{
|
|
ID: "pdfcpu",
|
|
New: func() gotenberg.Module { return new(PdfCpu) },
|
|
}
|
|
}
|
|
|
|
// Provision sets the engine properties.
|
|
func (engine *PdfCpu) Provision(ctx *gotenberg.Context) error {
|
|
binPath, ok := os.LookupEnv("PDFCPU_BIN_PATH")
|
|
if !ok {
|
|
return errors.New("PDFCPU_BIN_PATH environment variable is not set")
|
|
}
|
|
|
|
engine.binPath = binPath
|
|
|
|
return nil
|
|
}
|
|
|
|
// Validate validates the module properties.
|
|
func (engine *PdfCpu) Validate() error {
|
|
_, err := os.Stat(engine.binPath)
|
|
if os.IsNotExist(err) {
|
|
return fmt.Errorf("pdfcpu binary path does not exist: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Merge combines multiple PDFs into a single PDF.
|
|
func (engine *PdfCpu) Merge(ctx context.Context, logger *zap.Logger, inputPaths []string, outputPath string) error {
|
|
var args []string
|
|
args = append(args, "merge", outputPath)
|
|
args = append(args, inputPaths...)
|
|
|
|
cmd, err := gotenberg.CommandContext(ctx, logger, engine.binPath, args...)
|
|
if err != nil {
|
|
return fmt.Errorf("create command: %w", err)
|
|
}
|
|
|
|
_, err = cmd.Exec()
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("merge PDFs with pdfcpu: %w", err)
|
|
}
|
|
|
|
// Split splits a given PDF file.
|
|
func (engine *PdfCpu) Split(ctx context.Context, logger *zap.Logger, mode gotenberg.SplitMode, inputPath, outputDirPath string) ([]string, error) {
|
|
var args []string
|
|
|
|
switch mode.Mode {
|
|
case gotenberg.SplitModeIntervals:
|
|
args = append(args, "split", "-mode", "span", inputPath, outputDirPath, mode.Span)
|
|
case gotenberg.SplitModePages:
|
|
if mode.Unify {
|
|
outputPath := fmt.Sprintf("%s/%s", outputDirPath, filepath.Base(inputPath))
|
|
args = append(args, "trim", "-pages", mode.Span, inputPath, outputPath)
|
|
break
|
|
}
|
|
args = append(args, "extract", "-mode", "page", "-pages", mode.Span, inputPath, outputDirPath)
|
|
default:
|
|
return nil, fmt.Errorf("split PDFs using mode '%s' with pdfcpu: %w", mode.Mode, gotenberg.ErrPdfSplitModeNotSupported)
|
|
}
|
|
|
|
cmd, err := gotenberg.CommandContext(ctx, logger, engine.binPath, args...)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("create command: %w", err)
|
|
}
|
|
|
|
_, err = cmd.Exec()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("split PDFs with pdfcpu: %w", err)
|
|
}
|
|
|
|
outputPaths, err := gotenberg.WalkDir(outputDirPath, ".pdf")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("walk directory to find resulting PDFs from split with pdfcpu: %w", err)
|
|
}
|
|
|
|
return outputPaths, nil
|
|
}
|
|
|
|
// Flatten is not available in this implementation.
|
|
func (engine *PdfCpu) Flatten(ctx context.Context, logger *zap.Logger, inputPath string) error {
|
|
return fmt.Errorf("flatten PDF with pdfcpu: %w", gotenberg.ErrPdfEngineMethodNotSupported)
|
|
}
|
|
|
|
// Convert is not available in this implementation.
|
|
func (engine *PdfCpu) Convert(ctx context.Context, logger *zap.Logger, formats gotenberg.PdfFormats, inputPath, outputPath string) error {
|
|
return fmt.Errorf("convert PDF to '%+v' with pdfcpu: %w", formats, gotenberg.ErrPdfEngineMethodNotSupported)
|
|
}
|
|
|
|
// ReadMetadata is not available in this implementation.
|
|
func (engine *PdfCpu) ReadMetadata(ctx context.Context, logger *zap.Logger, inputPath string) (map[string]interface{}, error) {
|
|
return nil, fmt.Errorf("read PDF metadata with pdfcpu: %w", gotenberg.ErrPdfEngineMethodNotSupported)
|
|
}
|
|
|
|
// WriteMetadata is not available in this implementation.
|
|
func (engine *PdfCpu) WriteMetadata(ctx context.Context, logger *zap.Logger, metadata map[string]interface{}, inputPath string) error {
|
|
return fmt.Errorf("write PDF metadata with pdfcpu: %w", gotenberg.ErrPdfEngineMethodNotSupported)
|
|
}
|
|
|
|
// Interface guards.
|
|
var (
|
|
_ gotenberg.Module = (*PdfCpu)(nil)
|
|
_ gotenberg.Provisioner = (*PdfCpu)(nil)
|
|
_ gotenberg.Validator = (*PdfCpu)(nil)
|
|
_ gotenberg.PdfEngine = (*PdfCpu)(nil)
|
|
)
|