Jython+Selenium2Library無法在Windows上啟動Firefox

這陣子開始著手於升級Selenium2,我們的環境是透過Java啟動RobotFramework(2.9),在Runtime時去編譯Selenium與Selenium2Library。這樣的模式在Linux上,可以正常的跑完測試用的測試案例;然而在Windows上卻會出現如下圖錯誤:

首先會發現firefox並沒被啟動,然後有OSError: Operation not permitted。後來花了一些時間trace了Selenium的程式碼,發現錯誤是在複製firefox profile到暫存資料夾產生的:

# firefox_profile.py的__init__
    shutil.copytree(self.profile_dir, newprof,
          ignore=shutil.ignore_patterns("parent.lock", "lock", ".parentlock"))
雖然檔案都有被複製完成,但就是會發生錯誤;雖然嘗試將例外catch起來吃掉,但不只這地方會發生問題。於是我就開始找尋原因。我將robotframework.jar中Lib/shutil.py.class刪除,並將shutil.py複製到Lib資料夾下:


後來發現問題是在copystat>copystat做hasattr(os, 'chflags')的動作。原本程式碼中,有針對Windows發生例外(WindowsError)時,會做pass的動作:
    try:
        copystat(src, dst)
    except OSError, why:
        if WindowsError is not None and isinstance(why, WindowsError):
            # Copying file access times may fail on Windows
            pass
        else:
            errors.append((src, dst, str(why)))
然而在這地方所收到的例外卻是一般OSError,我在這地方特別印出os.name,想確認runtime所抓到的是否為windows,沒想到卻是java。也許這是造成無法正確丟出WindowsError的原因之一。於是我想在生問題時,如果是在Windows上,就直接pass好了,之後有新的robotframework再來看看。最後我這樣修改,暫時躲掉了這問題:
    try:
        copystat(src, dst)
    except OSError, why:
        if WindowsError is not None and isinstance(why, WindowsError):
            # Copying file access times may fail on Windows
            pass
        elif iswindows():
            pass        
        else:
            errors.append((src, dst, str(why)))
    if errors:
        raise Error, errors
 
def iswindows():
    ver = sys.platform.lower()
    if ver.startswith('java'):
        import java.lang 
        os = java.lang.System.getProperty( "os.name" )
        return "win" in os.lower()		
    return "win" in ver

友藏內心獨白: 有沒有無痛升級祕方阿?

  • 2015-11-26 如果Firefox非安裝在預設路徑,也可能有問題,可以參考這篇