質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.77%

  • C#

    6579questions

    C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

  • Chrome

    567questions

    Google Chromeは携帯、テレビ、デスクトップなどの様々なプラットフォームで利用できるウェブブラウザです。Googleが開発したもので、Blink (レンダリングエンジン) とアプリケーションフレームワークを使用しています。

  • selenium

    461questions

    Selenium(セレニウム)は、ブラウザをプログラムで作動させるフレームワークです。この原理を使うことにより、ブラウザのユーザーテストなどを自動化にすることができます。

C#+SeleniumでStaleElementReferenceError

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 220

masahiro.o

score 5

C#でSeleniumを使用して自動化処理を書いています。
(ブラウザはChromeです。)

OS: Windows10
IDE: Visual Studio 2015
Selenium: 3.11.2
ChromeDriver: 2.37.0

必ずではないのですが、頻繁に「stale element reference: element is not attached to the page document」という
StaleElementReferenceErrorが発生して困っています。

例えば、特定のaタグをクリックする処理で、DisplayedもtrueでTextの値も取れる状態でClickすると発生したりします。

色々調べたのですが、完全に描画されるまで待つしかなさそうで
以下のようにWebDriverWaitで待ってみたりしても不安定です…

        private static IWebElement FindElementEx(By by, uint timeOutInSeconds)
        {
            if (timeOutInSeconds > 0)
            {
                var wait = new WebDriverWait(webDriver, TimeSpan.FromSeconds(timeOutInSeconds));
                return wait.Until<IWebElement>(ctx => ctx.FindElement(by));
            }
            return webDriver.FindElement(by);
        }

仕方なく上の処理で要素を取得して、Sleep(1000)などとして回避していますが
もっとスマートなやり方はございませんでしょうか。

知っている方がいらっしゃればご教示お願いいたします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

check解決した方法

0

stack over flowにそれっぽいのがあったので一先ず解決にします。
(これでもダメなときはダメですがマシにはなりました。)
需要があるかは分かりませんが一応、共有しておきます。

以下の拡張クラスをプロジェクトに追加します。

        /// <summary>
        ///     Method that finds an element based on the search parameters within a specified timeout.
        /// </summary>
        /// <param name="context">The context where this is searched. Required for extension methods</param>
        /// <param name="by">The search parameters that are used to identify the element</param>
        /// <param name="timeOutInSeconds">The time that the tool should wait before throwing an exception</param>
        /// <returns> The first element found that matches the condition specified</returns>
        public static IWebElement FindElement(this ISearchContext context, By by, uint timeOutInSeconds)
        {
            if (timeOutInSeconds > 0)
            {
                var wait = new DefaultWait<ISearchContext>(context);
                wait.Timeout = TimeSpan.FromSeconds(timeOutInSeconds);
                return wait.Until<IWebElement>(ctx => ctx.FindElement(by));
            }
            return context.FindElement(by);
        }
        /// <summary>
        ///     Method that finds a list of elements based on the search parameters within a specified timeout.
        /// </summary>
        /// <param name="context">The context where this is searched. Required for extension methods</param>
        /// <param name="by">The search parameters that are used to identify the element</param>
        /// <param name="timeoutInSeconds">The time that the tool should wait before throwing an exception</param>
        /// <returns>A list of all the web elements that match the condition specified</returns>
        public static IReadOnlyCollection<IWebElement> FindElements(this ISearchContext context, By by, uint timeoutInSeconds)
        {

            if (timeoutInSeconds > 0)
            {
                var wait = new DefaultWait<ISearchContext>(context);
                wait.Timeout = TimeSpan.FromSeconds(timeoutInSeconds);
                return wait.Until<IReadOnlyCollection<IWebElement>>(ctx => ctx.FindElements(by));
            }
            return context.FindElements(by);
        }
        /// <summary>
        ///     Method that finds a list of elements with the minimum amount specified based on the search parameters within a specified timeout.<br/>
        /// </summary>
        /// <param name="context">The context where this is searched. Required for extension methods</param>
        /// <param name="by">The search parameters that are used to identify the element</param>
        /// <param name="timeoutInSeconds">The time that the tool should wait before throwing an exception</param>
        /// <param name="minNumberOfElements">
        ///     The minimum number of elements that should meet the criteria before returning the list <para/>
        ///     If this number is not met, an exception will be thrown and no elements will be returned
        ///     even if some did meet the criteria
        /// </param>
        /// <returns>A list of all the web elements that match the condition specified</returns>
        public static IReadOnlyCollection<IWebElement> FindElements(this ISearchContext context, By by, uint timeoutInSeconds, int minNumberOfElements)
        {
            var wait = new DefaultWait<ISearchContext>(context);
            if (timeoutInSeconds > 0)
            {
                wait.Timeout = TimeSpan.FromSeconds(timeoutInSeconds);
            }

            // Wait until the current context found the minimum number of elements. If not found after timeout, an exception is thrown
            wait.Until<bool>(ctx => ctx.FindElements(by).Count >= minNumberOfElements);

            //If the elements were successfuly found, just return the list
            return context.FindElements(by);
        }

使い方は以下のような感じです。

ar driver = new FirefoxDriver();
driver.Navigate().GoToUrl("http://localhost");
var main = driver.FindElement(By.Id("main"));
// It can be now used to wait when using elements to search
var btn = main.FindElement(By.Id("button"),10);
btn.Click();
//This will wait up to 10 seconds until a button is found
var button = driver.FindElement(By.TagName("button"),10)
//This will wait up to 10 seconds until a button is found, and return all the buttons found
var buttonList = driver.FindElements(By.TagName("button"),10)
//This will wait for 10 seconds until we find at least 5 buttons
var buttonsMin= driver.FindElements(By.TagName("button"), 10, 5);
driver.Close();

色々やったのですが、上の対応を行いまだエラーが出る場合はSleepを少し入れるなどの対応で回避しています。
あまり試せていませんが、ボタンやリンクなどのクリックはJavaScriptでクリックするように仕込んだ方が安定するかもしれません。

C#でSelenium触ってる人は少ないんですかね…日本語の情報少ないです。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.77%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • C#

    6579questions

    C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

  • Chrome

    567questions

    Google Chromeは携帯、テレビ、デスクトップなどの様々なプラットフォームで利用できるウェブブラウザです。Googleが開発したもので、Blink (レンダリングエンジン) とアプリケーションフレームワークを使用しています。

  • selenium

    461questions

    Selenium(セレニウム)は、ブラウザをプログラムで作動させるフレームワークです。この原理を使うことにより、ブラウザのユーザーテストなどを自動化にすることができます。