Skip to content

Python binding for the Sane API (using ctypes) and abstration layers

License

Notifications You must be signed in to change notification settings

gariels/pyinsane

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

PyInsane 2

Description

Python library to access and use image scanners.

Support for:

  • Sane (Scanners on GNU/Linux, *BSD, MacOSX, etc)
  • WIA 2 (Windows Image Acquisition ; Scanners on Microsoft Windows >= Vista)

It supports:

  • Flatbed
  • Automatic Document Feeder
  • While scanning, can provide chunks of the image for on-the-fly preview (see Paperwork for instance)
  • Python 2.7 (GNU/Linux only)
  • Python 3.x (GNU/Linux and Windows)

Not tested but should work too:

  • Handheld image scanners

Dependencies

On all platforms:

  • Pillow (if the abstraction layer is used)

Platform specific:

  • GNU/Linux, *BSD, MacOSX, etc: libsane

Supported scanners

In theory, all scanners supported by Sane or WIA should work.

In practice, each driver tends to have its own quirks. Pyinsane tries to include most of the workarounds you may need.

There is a list of scanners known to work (or to have worked at some point).

Installation

# recommanded to get the latest stable version
sudo pip3 install pyinsane2

or

# for the development version
git clone https://github.com/openpaperwork/pyinsane.git
cd pyinsane
sudo make install  # will run 'python3 ./setup.py install'

Installation on GNU/Linux should work out-of-the-box.

Installation on Windows will require Python, Visual C++ and WinDDK (see below for details).

Tests

make check  # check style + static analysis
make test  # run tests

Tests require at least one scanner with a flatbed and an ADF (Automatic Document Feeder).

If possible, they should be run with at least 2 scanners connected. The first that appear in "scanimage -L" must be the one with the ADF.

For reference, my current setup is:

  • HP Officejet 4620 (Flatbed + ADF)
  • HP Deskjet 2050 J510 series (Flatbed)

On GNU/Linux, you can simply enable the Sane backend 'test'.

Usage

Scanner detection

import pyinsane2

pyinsane2.init()
try:
	devices = pyinsane2.get_devices()
	assert(len(devices) > 0)
	device = devices[0]

	print("I'm going to use the following scanner: %s" % (str(device)))
	scanner_id = device.name
finally:
	pyinsane2.exit()

or if you already know its name/id:

import pyinsane2

pyinsane2.init()
try:
	device = pyinsane2.Scanner(name="somethingsomething")
	print("I'm going to use the following scanner: %s" % (str(device)))
finally:
	pyinsane2.exit()

Simple scan

import pyinsane2

pyinsane2.init()
try:
	devices = pyinsane2.get_devices()
	assert(len(devices) > 0)
	device = devices[0]
	print("I'm going to use the following scanner: %s" % (str(device)))

	pyinsane2.set_scanner_opt(device, 'resolution', [300])

# Beware: Some scanners have "Lineart" or "Gray" as default mode
# better set the mode everytime
	pyinsane2.set_scanner_opt(device, 'mode', ['Color'])

# Beware: by default, some scanners only scan part of the area
# they could scan.
	pyinsane2.maximize_scan_area(device)

	scan_session = device.scan(multiple=False)
	try:
		while True:
			scan_session.scan.read()
	except EOFError:
		pass
	image = scan_session.images[-1]
finally:
	pyinsane2.exit()

See examples/scan.py for a more complete example.

Multiple scans using an automatic document feeder (ADF)

import pyinsane2

pyinsane2.init()
try:
	devices = pyinsane2.get_devices()
	assert(len(devices) > 0)
	device = devices[0]
	print("I'm going to use the following scanner: %s" % (str(device)))

	try:
		pyinsane2.set_scanner_opt(device, 'source', ['ADF', 'Feeder'])
	except PyinsaneException:
		print("No document feeder found")
		return

# Beware: Some scanners have "Lineart" or "Gray" as default mode
# better set the mode everytime
	pyinsane2.set_scanner_opt(device, 'mode', ['Color'])

# Beware: by default, some scanners only scan part of the area
# they could scan.
	pyinsane2.maximize_scan_area(device)

	scan_session = device.scan(multiple=True)
	try:
		while True:
			try:
				scan_session.scan.read()
			except EOFError:
				print ("Got a page ! (current number of pages read: %d)"
					% (len(scan_session.images)))
	except StopIteration:
		print("Document feeder is now empty. Got %d pages"
			% len(scan_session.images))
	for idx in range(0, len(scan_session.images)):
		image = scan_session.images[idx]
finally:
	pyinsane2.exit()

Scanner's options

The options available depends on the backend and on the specific driver used.

The WIA implementation emulates common Sane options ('tl-x', 'br-x', 'tl-y', 'br-y', 'color', 'mode', 'source'). So you should use Sane options by default.

See the Sane documentation for the most common options.

Beware options casing can change between WIA and Sane implementation ! You should use pyinsane2.set_scanner_opt() whenever possible.

You can access the option values with:

device.options['option_name'].value

You can set the option values with:

device.options['option_name'].value = new_value
# or use the helper:
pyinsane2.set_scanner_opt(
    device, 'option_name',
    ['possible_new_value_1', 'possible_new_value_2']
)

You can get the constraint (accepted values) with:

device.options['option_name'].constraint

Constraints are usually:

  • None : unknown constraints / no constraint
  • tuple : (min_value, max_value)
  • list : possible values (Ex: ['Flatbed', 'Feeder'] or [75, 150, 300])

Note regarding the Sane implementation

When using the Sane API as is, some issues with some Sane drivers can become obvious in complex programs (uninitialized memory bytes, segfault, etc). You can get corrupted images or even crash your program.

This module works around issues like the following one by using a dedicated process for scanning:

corrupted scan --> scan fine

(see this comment for details)

When pyinsane2.init() is called, it will create 2 Unix pipes (FIFO) in your temporary directory and a dedicated process. To avoid forking other file descriptors from your program, you should initialize pyinsane2 as soon as possible.

Building requires nothing except Python. Libsane is loaded dynamically using ctypes.

Note regarding the WIA 2 implementation

Build

Build requires:

  • Either Python 3.4 + Windows SDK 7.1 (Visual C++ 2010) + Windows DDK (aka WDK)
  • Or Python 3.5 + Visual C++ 2016 + Windows DDK (aka WDK) (included in Visual Studio 2016)

(see the Python wiki for more information)

You must define the following environment values before calling python setup.py install:

  • WINDDK_INCLUDE_DIR (default value: c:\winddk\7600.16385.1\inc\atl71)
  • WINDDK_LIB_DIR (default value: c:\winddk\7600.16385.1\lib\ATL\amd64)

Usage

WIA provides one WiaItem2 by possible source (Flatbed, ADF, etc). And each of these items has its own properties.

To make the model consistent with Sane, all the properties have been merged in the same list. When you update a property specific to sources, it is updated on all the WIAItem2. The update is considered successful if it worked at least on one.

Some properties are emulated to make the API behavior consistent with the Sane implementation. The WIA implementation emulates common Sane options ('tl-x', 'br-x', 'tl-y', 'br-y', 'color', 'mode', 'source'). If your program must be cross-platform, you are strongly advised to use these emulated options instead of the WIA ones ('xpos', 'ypos', 'xextent', 'yextent', etc).

Other examples

The folder 'examples' contains more detailed examples. For instance, examples/scan.py shows how to get pieces of a scan as it goes.

To run one of these scripts, run:

python -m examples.[script] [args]

For instance

python -m examples.scan toto.png

Contact

Application that uses Pyinsane

If you know of any other applications that use Pyinsane, please tell us :-)

Licence

GPL v3 2012-2016(c) Jerome Flesch ([email protected])

About

Python binding for the Sane API (using ctypes) and abstration layers

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Python 58.1%
  • C++ 39.7%
  • C 1.3%
  • Makefile 0.9%