#!env python3 # Eryn Wells ''' A Sudoku puzzle solver. ''' import itertools import math class Sudoku: def __init__(self, size=9, board=None): dim = math.sqrt(size) if dim != int(dim): raise ValueError('Board size must have an integral square root.') self._dimension = int(dim) self.size = size if board: self._board = bytearray(board)[:size**2] else: self._board = bytearray(b'\x00' * (size * size)) @property def dimension(self): return self._dimension @property def rows(self): return self._apply_index_ranges(self.index_rows) @property def columns(self): return self._apply_index_ranges(self.index_columns) @property def squares(self): return self._apply_index_ranges(self.index_squares) @property def index_rows(self): ''' Return an iterable of ranges of indexes into the board, each defining a row. ''' return (self._row(i, self.size) for i in range(self.size)) @property def index_columns(self): ''' Return an iterable of ranges of indexes into the board, each defining a column. ''' sz = self.size sz2 = sz ** 2 return (self._column(i, sz, sz2) for i in range(sz)) @property def index_squares(self): ''' Return an iterable of ranges of indexes into the board, each defining a square. ''' dim = self.dimension return (self._square(x, y, dim) for y in range(dim) for x in range(dim)) def peers(self, x, y): return {self._board[i] for i in self.index_peers(x, y) if self._board[i] != 0} def index_peers(self, x, y): dim = self.dimension sz = self.size sz2 = sz ** 2 sq_x, sq_y = int(x / dim), int(y / dim) return set(self._row(y, sz)) | set(self._column(x, sz, sz2)) | set(self._square(sq_x, sq_y, dim)) # TODO: Break the above into helper methods that produce a single thing given an index. def _row(self, r, size): return range(r * size, r * size + size) def _column(self, c, size, size2): return range(c, size2, size) def _square(self, x, y, dim): if (x < 0 or x >= dim) or (y < 0 or y >= dim): raise IndexError('Invalid coordinates for square: ({}, {})'.format(x, y)) offset = (x * dim, y * dim * self.size) def _range(i): start = (offset[1] + i * self.size) + offset[0] return range(start, start + dim) ranges = itertools.chain(*[_range(i) for i in range(dim)]) return ranges @property def solved(self): expected = set(range(1, self.size + 1)) all_groups = itertools.chain(self.rows, self.columns, self.squares) return all(expected == set(g) for g in all_groups) def _apply_index_ranges(self, ranges): return ((self._board[i] for i in r) for r in ranges) def __str__(self): field_width = len(str(self.size)) dim = self.dimension lines = [] spacer = '{0}{1}{0}'.format('+', '+'.join(['-' * (field_width * dim) for _ in range(dim)])) for line in range(self.size): chunks = [] for i in range(dim): fields = [] for j in range(dim): idx = line * self.size + i * dim + j fields.append('{{board[{i}]:^{width}}}'.format(i=idx, width=field_width)) chunks.append(''.join(fields)) if (line % dim) == 0: lines.append(spacer) lines.append('{0}{1}{0}'.format('|', '|'.join(chunks))) lines.append(spacer) fmt = '\n'.join(lines) str_board = [str(n) if n != 0 else ' ' for n in self._board] out = fmt.format(board=str_board) return out