Search code examples
pythonopenslide

How to merge tiles obtained using openslide-python


I am trying to combine tiles in the correct order so they end up as the same whole slide image (.svs file).

The .svs file is read from a filepath according to the function beloew:

def open_slide(filepath = None):
    try:
        slide = openslide.open_slide(filepath)
       
    except OpenSlideError as o:
        print("Error" + str(o))
        
        slide = None
    except FileNotFoundError as f:
        print("Error" +  str(f))
        slide = None
    return slide

In the picture below I am trying the merge the tiles I got using openslide-python's DeepZoom generator (see code snippet below)

def create_tile_generator(slide, tile_size, overlap):
     gen = DeepZoomGenerator(slide, tile_size=tile_size, overlap=overlap, limit_bounds=False)

This is how I split the .svs into its tiles:

def split_wsi_to_tiles(wsi_path = None):
    print("splitting wsi into tiles")
    tile_indices = process_slide(slide_num = SLIDE_NUM , filepath= wsi_path, tile_size = TILE_SIZE, overlap = OVERLAP)
    i = 0
    tile_indices_savepath  = os.path.join(os.getcwd(),"saved","tile_indices")
    save_file(filepath = tile_indices_savepath,filename=name,file= tile_indices)
    for ti in tile_indices:
        suffix =  str(i)
        (slide_num,tile) = process_tile_index(tile_index =ti,filepath = svs_path )
        tile = cv2.cvtColor(tile, cv2.COLOR_BGR2RGB)
        cv2.imwrite(save_path + suffix + ext,tile)
        i = i + 1
    print("done splitting wsi into tiles")
    return tile_indices_savepath

The helper functions process_slide and process_tile_index are given below

def process_slide(slide_num =1 , filepath= None, tile_size = 256, overlap = 0):
    slide = open_slide(filepath = filepath)
    generator = create_tile_generator(slide, tile_size, overlap)
    zoom_level = get_40x_zoom_level(slide, generator)
    print("zoom level set to " + str(zoom_level))
    cols, rows = generator.level_tiles[zoom_level - 1]
    tile_indices = [(slide_num, tile_size, overlap, zoom_level, col, row)
                  for col in range(cols) for row in range(rows)]
    return tile_indices

def process_tile_index(tile_index=None,filepath= None):
        slide_num, tile_size, overlap, zoom_level, col, row = tile_index
        slide = open_slide(filepath = filepath)
        generator = create_tile_generator(slide, tile_size, overlap)
        tile = np.asarray(generator.get_tile(zoom_level, (col, row)))
        return (slide_num, tile)

The get_40x_zoom_level function description:

def get_40x_zoom_level(slide, generator):
    global level
    highest_zoom_level = generator.level_count - 1  # 0-based indexing
    try:
        mag = int(slide.properties[openslide.PROPERTY_NAME_OBJECTIVE_POWER])
        
        offset = math.floor((mag / 40) / 2)
        level = highest_zoom_level - offset
    except (ValueError, KeyError) as e:
        level = highest_zoom_level
    print("zoom level set at " +str(level) )
    save_file(filepath= os.path.join(os.getcwd(),"saved"),filename = "level.pickle",file = level)
    return level

This is how I try to merge the tiles back to its whole slide image (not necessarily in .svs format but the same image):

def merge_tiles_to_wsi(tile_path= None,wsi_path = None):
    print("merging tiles into wsi")
    tile_indices = load_file(filepath = tile_indices_savepath,filename = name)
    slide = open_slide(filepath = wsi_path)

    level = load_file(filepath= os.path.join(os.getcwd(),"saved"),filename = "level.pickle")
    generator = create_tile_generator(slide, TILE_SIZE, OVERLAP)
    slide_dims = generator.level_dimensions[level]
    row_size = slide_dims[0]
    col_size = slide_dims[1]
    channel_size = 3
    slide_shape = (row_size,col_size,channel_size)
    print("shape of slide is " + str(slide_shape))
    wsi = np.zeros(slide_shape)
    for ti in tile_indices:
        slide_num, tile_size, overlap, zoom_level, col, row  = ti
        generator = create_tile_generator(slide, tile_size, overlap)
        tile = np.asarray(generator.get_tile(zoom_level, (col, row)))
        
        row_length = tile.shape[0]
        col_length = tile.shape[1]
        row_end = row + row_length
        col_end = col + col_length
        print("col: " + str(col) + " row: " + str(row) + str(wsi[row:row_end,col:col_end].shape) + " " + str(tile.shape))
        wsi[row:row_end,col:col_end] = tile
        # view_image(img= wsi)
    print("merging tiles into wsi")

    return wsi

Here is what the final output looks like out.png


Solution

  • libvips can do this merge and join for you. You can call it from pyvips, the Python binding.

    To load an svs image and split it into tiles you can write:

    import pyvips
    
    image = pyvips.Image.new_from_file("my-slide.svs")
    image.dzsave("my-deepzoom")
    

    And it'll write my-deepzoom.dzi and a directory, my-deepzoom_files, containing all the tiles. There are a lot of parameters you can adjust, see the chapter in the docs:

    https://libvips.github.io/libvips/API/current/Making-image-pyramids.md.html

    It's very fast and can make pyramids of any size on even modest hardware.

    You can recombine tiles to form images with arrayjoin. You give it a list of images in row-major order and set across to the number of images per row. For example:

    import pyvips
    
    tiles = [pyvips.Image.new_from_file(f"{x}_{y}.jpeg", access="sequential")
             for y in range(height) for x in range(width)] 
    image = pyvips.Image.arrayjoin(tiles, across=width)
    image.write_to_file("huge.tif", compression="jpeg", tile=True)
    

    It's very fast and can join extremely large arrays of images.