Read Excel cell value and not the formula computing it -openpyxl - python
I am using openpyxl to read cell value (excel addin-webservice update this column.
)
I have used data_only = True but it is not showing the current cell value instead it is the value stored the last time Excel read the sheet.
wbFile = openpyxl.load_workbook(filename = xxxx,data_only=True)
wsFile = wbFile[c_sSheet]
How can i read the cell actual value ?
wb = openpyxl.load_workbook(filename, data_only=True)
The data_only flag helps.
As #alex-martelli says, openpyxl does not evaluate formulae. When you open an Excel file with openpyxl you have the choice either to read the formulae or the last calculated value. If, as you indicate, the formula is dependent upon add-ins then the cached value can never be accurate. As add-ins outside the file specification they will never be supported. Instead you might want to look at something like xlwings which can interact with the Excel runtime.
data_only : read values for even for the formula cells.
keep_vba: it's used only if you are using macroenabled excel
file_location = 'C:\Arpan Saini\Monsters\Project_Testing\SecCardGrad\SecCardGrad_Latest_docs\Derived_Test_Cases_Secure_Card_Graduate.xlsm'
wb = load_workbook(file_location, keep_vba=True, data_only=True)
As #Charlie Clark mentioned you could use xlwings (if you have MS Excel). Here an example
say you have an excel sheet with formulas, for the example I define one with openpyxl
from openpyxl import Workbook, load_workbook
wb=Workbook()
ws1=wb['Sheet']
ws1['A1']='a'
ws1['A2']='b'
ws1['A3']='c'
ws1['B1']=1
ws1['B2']=2
ws1['B3']='=B1+B2'
wb.save('to_erase.xlsx')
As mentioned, if we load the excel again with openpyxl, we will not get the evaluated formula
wb2 = load_workbook(filename='to_erase.xlsx',data_only=True)
wb2['Sheet']['B3'].value
you can use xlwings to get the formula evaluated by excel:
import xlwings as xw
wbxl=xw.Book('to_erase.xlsx')
wbxl.sheets['Sheet'].range('B3').value
which returns 3, the expected value.
I found it quite useful when working with spreadsheets with very complicated formulas and references between sheets.
Faced the same problem. Needed to read cell values whatever those cells are: scalars, formulae with precomputed values or formulae without them, with fail-tolerance preferred over correctness.
The strategy is pretty straightforward:
if a cell doesn't contain formula, return cell's value;
if it's a formula, try to get its precomputed value;
if couldn't, try to evaluate it using pycel;
if failed (due to pycel's limited support of formulae or with some error), warn and return None.
I made a class which hides all this machinery and provides simple interface for reading cell values.
It's easy to modify the class so that it will raise an exception on step 4, if correctness is preferred over fail-tolerance.
Hope it will help someone.
from traceback import format_exc
from pathlib import Path
from openpyxl import load_workbook
from pycel.excelcompiler import ExcelCompiler
import logging
class MESSAGES:
CANT_EVALUATE_CELL = ("Couldn't evaluate cell {address}."
" Try to load and save xlsx file.")
class XLSXReader:
"""
Provides (almost) universal interface to read xlsx file cell values.
For formulae, tries to get their precomputed values or, if none,
to evaluate them.
"""
# Interface.
def __init__(self, path: Path):
self.__path = path
self.__book = load_workbook(self.__path, data_only=False)
def get_cell_value(self, address: str, sheet: str = None):
# If no sheet given, work with active one.
if sheet is None:
sheet = self.__book.active.title
# If cell doesn't contain a formula, return cell value.
if not self.__cell_contains_formula(address, sheet):
return self.__get_as_is(address, sheet)
# If cell contains formula:
# If there's precomputed value of the cell, return it.
precomputed_value = self.__get_precomputed(address, sheet)
if precomputed_value is not None:
return precomputed_value
# If not, try to compute its value from the formula and return it.
# If failed, report an error and return empty value.
try:
computed_value = self.__compute(address, sheet)
except:
logging.warning(MESSAGES.CANT_EVALUATE_CELL
.format(address=address))
logging.debug(format_exc())
return None
return computed_value
# Private part.
def __cell_contains_formula(self, address, sheet):
cell = self.__book[sheet][address]
return cell.data_type is cell.TYPE_FORMULA
def __get_as_is(self, address, sheet):
# Return cell value.
return self.__book[sheet][address].value
def __get_precomputed(self, address, sheet):
# If the sheet is not loaded yet, load it.
if not hasattr(self, '__book_with_precomputed_values'):
self.__book_with_precomputed_values = load_workbook(
self.__path, data_only=True)
# Return precomputed value.
return self.__book_with_precomputed_values[sheet][address].value
def __compute(self, address, sheet):
# If the computation engine is not created yet, create it.
if not hasattr(self, '__formulae_calculator'):
self.__formulae_calculator = ExcelCompiler(self.__path)
# Compute cell value.
computation_graph = self.__formulae_calculator.gen_graph(
address, sheet=sheet)
return computation_graph.evaluate(f"{sheet}!{address}")
I solved this problem by the following way:
import xlwings
from openpyxl import load_workbook
data = load_workbook('PATH_TO_YOUR_XLSX_FILE')
data['sheet_name']['A1'].value = 1
data.save('PATH_TO_YOUR_XLSX_FILE')
excel_app = xlwings.App(visible=False)
excel_book = excel_app.books.open('PATH_TO_YOUR_XLSX_FILE')
excel_book.save()
excel_book.close()
excel_app.quit()
data = load_workbook('PATH_TO_YOUR_XLSX_FILE', data_only=True)
I hope, this can help You...
Instead on openpyxl, use xlwings.
I found data_only option is not working properly if there is an "REF!" error cell in a worksheet.
Openpyxl returns None for each cell value in my tiny test xlsx file.
For me, after opening Excel and fixing the cell, data_only works perfectly.
I use openpyxl 3.0.3
Rather than use a Python library to do the Excel calculations, I have Excel do them.
Why? It's not pure Python, but it minimizes the amount of Python involved. Instead of using Python to evaluate the Excel formulas, I let Excel handle its own functionality. This avoids any possible bugs in the Python that evaluates the Excel formulas.
Here's an outline of how this approach works:
Call openpyxl with data_only=False to edit and then save the spreadsheet.
Use subprocess.Popen to open the new spreadsheet in Excel, and let Excel evaluate the spreadsheet formulas.
Use pynput.keyboard to save the updated spreadsheet and exit Excel.
Use openpyxl with data_only=True to open the updated spreadsheet and get the values of the formulas.
Here is a test program for Windows that creates a new workbook, puts the formula "=SUM(Al:C3)" in cell E2, puts data into cells A1-C3, and evaluates the formula.
from openpyxl import load_workbook, Workbook
from pynput.keyboard import Key, Controller
import subprocess
import time
import os
excel_prog = r'C:\Program Files\Microsoft Office\root\Office16\EXCEL.EXE'
# Create test Excel workbook, get default worksheet.
wb = Workbook()
ws = wb.active
# Put data and a formula into worksheet.
for row_index in range(1,4):
for column_index in range(1,4):
ws.cell(row = row_index, column = column_index).value = row_index + column_index
ws['E1'].value = 'Sum of cells in range A1:C3:'
ws['E2'].value = '=SUM(A1:C3)'
# Try to get value of formula. We'll see the formula instead.
print('E2:', ws['E2'].value)
# Save and close workbook.
wb.save(filename = 'test.xlsx')
wb.close()
# Pause to give workbook time to close.
time.sleep(5)
# Open the workbook in Excel. I specify folder, otherwise Excel will
# open in "Protected View", interfering with using pynput.
subprocess.Popen([excel_prog, os.path.join(os.getcwd(), 'test.xlsx')])
# Pause to give workbook time to open and for formulas to update.
time.sleep(5)
# Save workbook using pynput.
keyboard = Controller()
with keyboard.pressed(Key.ctrl):
keyboard.press('s')
keyboard.release('s')
# Pause to give workbook time to save.
time.sleep(5)
# Close workbook.
with keyboard.pressed(Key.alt):
keyboard.press(Key.f4)
keyboard.release(Key.f4)
# Pause to give workbook time to fully close.
time.sleep(5)
# Open Excel workbook and worksheet in openpyxl, data-only.
wb = load_workbook(filename = 'test.xlsx', data_only = True)
ws = wb.active
# Get value of the cell containing the formula.
print('E2:', ws['E2'].value)
# Close workbook.
wb.close()
Xlcalculator has the ability to evaluate a cell.
from xlcalculator import ModelCompiler
from xlcalculator import Model
from xlcalculator import Evaluator
filename = r'xxxx.xlsm'
compiler = ModelCompiler()
new_model = compiler.read_and_parse_archive(filename)
evaluator = Evaluator(new_model)
val1 = evaluator.evaluate('First!A2')
print("value 'evaluated' for First!A2:", val1)
The output is:
value 'evaluated' for First!A2: 0.1
Related
Is there a way to save data in named Excel cells using Python?
I have used openpyxl for outputting values in Excel in my Python code. However, now I find myself in a situation where the cell locations in excel file may change based on the user. To avoid any problems with the program, I want to name the cells where the code can save the output to. Is there any way to have Python interact with named ranges in Excel?
For a workbook level defined name import openpyxl wb = openpyxl.load_workbook("c:/tmp/SO/namerange.xlsx") ws = wb["Sheet1"] mycell = wb.defined_names['mycell'] for title, coord in mycell.destinations: ws = wb[title] ws[coord] = "Update" wb.save('updated.xlsx') print("{} {} updated".format(ws,coord))
I was able to find the parameters of the named range using defined_names. After that I just worked like it was a normal Excel cell. from openpyxl import load_workbook openWB=load_workbook('test.xlsx') rangeDestination = openWB.defined_names['testCell'] print(rangeDestination) sheetName=str(rangeDestination.attr_text).split('!')[0] cellName = str(rangeDestination.attr_text).split('!')[1] sheetToWrite=openWB[sheetName] cellToWrite=sheetToWrite[cellName] sheetToWrite[cellName]='TEST-A3' print(sheetName) print(cellName) openWB.save('test.xlsx') openWB.close()
Openpyxl Reading Non-empty Cells as None [duplicate]
I have a simple excel file: A1 = 200 A2 = 300 A3 = =SUM(A1:A2) this file works in excel and shows proper value for SUM, but while using openpyxl module for python I cannot get value in data_only=True mode Python code from shell: wb = openpyxl.load_workbook('writeFormula.xlsx', data_only = True) sheet = wb.active sheet['A3'] <Cell Sheet.A3> # python response print(sheet['A3'].value) None # python response while: wb2 = openpyxl.load_workbook('writeFormula.xlsx') sheet2 = wb2.active sheet2['A3'].value '=SUM(A1:A2)' # python response Any suggestions what am I doing wrong?
It depends upon the provenance of the file. data_only=True depends upon the value of the formula being cached by an application like Excel. If, however, the file was created by openpyxl or a similar library, then it's probable that the formula was never evaluated and, thus, no cached value is available and openpyxl will report None as the value.
I have replicated the issue with Openpyxl and Python. I am currently using openpyxl version 2.6.3 and Python 3.7.4. Also I am assuming that you are trying to complete an exercise from ATBSWP by Al Sweigart. I tried and tested Charlie Clark's answer, considering that Excel may indeed cache values. I opened the spreadsheet in Excel, copied and pasted the formula into the same exact cell, and finally saved the workbook. Upon reopening the workbook in Python with Openpyxl with the data_only=True option, and reading the value of this cell, I saw the proper value, 500, instead of the wrong value, the None type. I hope this helps.
I had the same issue. This may not be the most elegant solution, but this is what worked for me: import xlwings from openpyxl import load_workbook excel_app = xlwings.App(visible=False) excel_book = excel_app.books.open('writeFormula.xlsx') excel_book.save() excel_book.close() excel_app.quit() workbook = load_workbook(filename='writeFormula.xlsx', data_only=True)
I have suggestion to this problem. Convert xlsx file to csv :). You will still have the original xlsx file. The conversion is done by libreoffice (it is that subprocess.call() line).You can use also Pandas for this as a more pythonic way. from subprocess import call from openpyxl import load_workbook from csv import reader filename="test" wb = load_workbook(filename+".xlsx") spread_range = wb['Sheet1'] #what ever function there is in A1 cell to be evaluated print(spread_range.cell(row=1,column=1).value) wb.close() #this line can be done with subprocess or os.system() #libreoffice --headless --convert-to csv $filename --outdir $outdir call("libreoffice --headless --convert-to csv "+filename+".xlsx", shell=True) with open(filename+".csv", newline='') as f: reader = reader(f) data = list(reader) print(data[0][0]) or # importing pandas as pd import pandas as pd # read an excel file and convert # into a dataframe object df = pd.DataFrame(pd.read_excel("Test.xlsx")) # show the dataframe df I hope this helps somebody :-)
Yes, #Beno is right. If you want to edit the file without touching it, you can make a little "robot" that edits your excel file. WARNING: This is a recursive way to edit the excel file. These libraries are depend on your machine, make sure you set time.sleep properly before continuing the rest of the code. For instance, I use time.sleep, subprocess.Popen, and pywinauto.keyboard.send_keys, just add random character to any cell that you set, then save it. Then the data_only=True is working perfectly. for more info about pywinauto.keyboard: pywinauto.keyboard # import these stuff import subprocess from pywinauto.keyboard import send_keys import time import pygetwindow as gw import pywinauto excel_path = r"C:\Program Files\Microsoft Office\root\Office16\EXCEL.EXE" excel_file_path = r"D:\test.xlsx" def focus_to_window(window_title=None): # function to focus to window. https://stackoverflow.com/a/65623513/8903813 window = gw.getWindowsWithTitle(window_title)[0] if not window.isActive: pywinauto.application.Application().connect(handle=window._hWnd).top_window().set_focus() subprocess.Popen([excel_path, excel_file_path]) time.sleep(1.5) # wait excel to open. Depends on your machine, set it propoerly focus_to_window("Excel") # focus to that opened file send_keys('%{F3}') # excel's name box | ALT+F3 send_keys('AA1{ENTER}') # whatever cell do you want to insert somthing | Type 'AA1' then press Enter send_keys('Stackoverflow.com') # put whatever you want | Type 'Stackoverflow.com' send_keys('^s') # save | CTRL+S send_keys('%{F4}') # exit | ALT+F4 print("Done") Sorry for my bad english.
As others already mentioned, Openpyxl only reads cashed formula value in data_only mode. I have used PyWin32 to open and save each XLSX file before it's processed by Openpyxl to read the formulas result value. This works for me well, as I don't process large files. This solution will work only if you have MS Excel installed on your PC. import os import win32com.client from openpyxl import load_workbook # Opening and saving XLSX file, so results for each stored formula can be evaluated and cashed so OpenPyXL can read them. excel_file = os.path.join(path, file) excel = win32com.client.gencache.EnsureDispatch('Excel.Application') excel.DisplayAlerts = False # disabling prompts to overwrite existing file excel.Workbooks.Open(excel_file ) excel.ActiveWorkbook.SaveAs(excel_file, FileFormat=51, ConflictResolution=2) excel.DisplayAlerts = True # enabling prompts excel.ActiveWorkbook.Close() wb = load_workbook(excel_file) # read your formula values with openpyxl and do other stuff here
I ran into the same issue. After reading through this thread I managed to fix it by simply opening the excel file, making a change then saving the file again. What a weird issue.
How to copy a sheet in one workbook and paste just the values in a new workbook?
I have an excel workbook that has quite a few formulas, and when I try to upload the workbook into a database, the cells with iferror formulas come in as blanks even though it should be a string or number. I am new to python but I want to create a python file that will read in the sheet, and paste only the values into a new workbook. I tried: import openpyxl as xl wb1 = xl.load_workbook('file1.xlsx') ws1 = wb1["Sheet 1"] wb2 = xl.load_workbook('file2.xlsx') ws2 = wb2.create_sheet(ws1.title) for row in ws1: for cell in row: ws2[cell.coordinate].value = cell.value wb2.save('path') The code works to copy the data into a new workbook, but it is pasting the formulas. I just want the values.
As per my earlier comment: This comes from the OpenPyxl docs: Where it's stated on the openpyxl.reader.excel.load_workbook submodule, looking at the data_only parameter: data_only (bool) – controls whether cells with formulae have either the formula (default) or the value stored the last time Excel read the sheet Default is Formulas whereas you want the values. So setting it to true: wb = xl.load_workbook('file1.xlsx', data_only=True) Should help :)
How to get the value from merged cells in xlsx file using python?
I am trying to get the value from cell with row = 11 and column B and C. See screenshot for more clarification. I tried following code using xlrd package but it does not print anything. import xlrd path = "C:/myfilepath/data.xlsx" workbook = xlrd.open_workbook(path) sheet = workbook.sheet_by_index(0) sheet.cell_value(10,1) sheet.cell_value(10,2) I am not able to output the value from particular merged cells using xlrd package in python. Above code should print the cell value i.e PCHGFT001KS
I don't know how xlrd works, but I do know how the lovely openpyxl works. You should use openpyxl! it's a robust tool for working with xlsx files. (NOT xls). import openpyxl wb = openpyxl.load_workbook(excel) ws = wb[wb.get_sheet_names()[0]] print(ws['B11'].value) Extra: If you want to unmerge those blocks you can do the following. for items in ws.merged_cell_ranges: ws.unmerge_cells(str(items)) wb.save(excel)
How get a excel sheet with its code name property with "python"
I want to get a Excel's sheet with Python. I can do this with the sheet's name but I want get it with its Code Name property. The following is a code using the sheet's name: from openpyxl import load_workbook wb_donnees = load_workbook("Données.xlsm", read_only = True) name_ws_1 = wb_donnees.get_sheet_name()[0] ws_1 = wb_donnees[name_ws_1] But I want get the sheet with its Code Name property. Is it possible ?
Charlie Clark's answer works for me in read mode. I'm not sure whether OP needed this, but when writing a new workbook, you cannot get the codename this way. Instead, you will need to specify it yourself, otherwise the function returns None, and sheets will only be codenamed 'Sheet1' etc at workbook creation. wb = load_workbook('input.xlsm') wsx = wb.create_sheet('New Worksheet') wsx.sheet_properties.codeName = 'wsx' wb.save('output.xlsm')
The following should will only work if the file is not opened in read-only mode: from openpyxl import load_workbook wb = load_workbook("Données.xlsm") for n in wb.sheetnames: ws = wb[n] print(n, ws.sheet_properties.codeName)