Очень медленные запросы MongoDB при использовании IXSCAN в качестве выигрышного плана для запросов

У меня есть БД с около 30M записей в коллекции размером около 100 ГБ (всего документов и индексов).

У меня есть составной индекс, который фильтрует данные на основе user_id и некоторых других полей, таких как: is_active, is_logged_in, так далее.

С MongoDB Compass Я вижу медленные запросы около 10 секунд, 20 секунд или даже 40 секунд! Я выполнил точно такой же запрос, и результат извлекается менее чем за 500 мс (хотя он может кэшироваться со второй попытки).

Когда я продолжу op stat, я вижу следующий статус блокировки:

 "lockStats": {
         "Global": {
             "acquireCount": {
                 "r": 574
             }
         },
         "MMAPV1Journal": {
             "acquireCount": {
                 "r": 295
             },
             "acquireWaitCount": {
                 "r": 2
             },
             "timeAcquiringMicros": {
                 "r": 15494
             }
         },
 }

acquireCount: Количество раз, когда операция получала блокировку в указанном режиме, настолько высоко по сравнению с быстрым запросом (в другой коллекции), который имеет следующий статус:

"lockStats": {
         "Global": {
             "acquireCount": {
                 "r": 2
             }
         },
         "MMAPV1Journal": {
             "acquireCount": {
                 "r": 1
             }
         },
         "Database": {
             "acquireCount": {
                 "r": 1
             }
         },
         "Collection": {
             "acquireCount": {
                 "R": 1
             }
         }
     }

Когда операции выполняются медленно, а у пользователя, имеющего много записей, так много времени, через несколько секунд эффект домино воздействует на все остальные операции.

Когда я объясняю запрос для большой коллекции, я вижу результат, который использовал индекс:

{
    "queryPlanner" : {
        "plannerVersion" : 1,
        "namespace" : "cuda.call_history",
        "indexFilterSet" : false,
        "parsedQuery" : {
            "$and" : [ 
                {
                    "$or" : [ 
                        {
                            "trk.0.extra_data.spam.is_spam" : {
                                "$eq" : false
                            }
                        }, 
                        {
                            "$nor" : [ 
                                {
                                    "trk.0.extra_data.spam.is_spam" : {
                                        "$exists" : true
                                    }
                                }
                            ]
                        }
                    ]
                }, 
                {
                    "is_removed" : {
                        "$eq" : false
                    }
                }, 
                {
                    "user_id" : {
                        "$eq" : "00000000000040008000000000002a5d"
                    }
                }, 
                {
                    "trk.0.direction" : {
                        "$eq" : "ingress"
                    }
                }, 
                {
                    "trk.0.type" : {
                        "$eq" : "fax"
                    }
                }, 
                {
                    "date" : {
                        "$lt" : "2018-01-09 10:36:31"
                    }
                }, 
                {
                    "date" : {
                        "$gt" : "1970-01-01 00:00:00"
                    }
                }, 
                {
                    "trk.0.data.status" : {
                        "$in" : [ 
                            "p_received", 
                            "success"
                        ]
                    }
                }
            ]
        },
        "winningPlan" : {
            "stage" : "FETCH",
            "filter" : {
                "$and" : [ 
                    {
                        "$or" : [ 
                            {
                                "trk.0.extra_data.spam.is_spam" : {
                                    "$eq" : false
                                }
                            }, 
                            {
                                "$nor" : [ 
                                    {
                                        "trk.0.extra_data.spam.is_spam" : {
                                            "$exists" : true
                                        }
                                    }
                                ]
                            }
                        ]
                    }, 
                    {
                        "trk.0.type" : {
                            "$eq" : "fax"
                        }
                    }, 
                    {
                        "date" : {
                            "$lt" : "2018-01-09 10:36:31"
                        }
                    }, 
                    {
                        "date" : {
                            "$gt" : "1970-01-01 00:00:00"
                        }
                    }
                ]
            },
            "inputStage" : {
                "stage" : "IXSCAN",
                "keyPattern" : {
                    "user_id" : 1,
                    "trk.0.direction" : 1,
                    "is_read" : 1,
                    "trk.0.data.status" : 1,
                    "is_removed" : 1
                },
                "indexName" : "user_id_direction_is_read_status_is_removed",
                "isMultiKey" : false,
                "isUnique" : false,
                "isSparse" : false,
                "isPartial" : false,
                "indexVersion" : 1,
                "direction" : "forward",
                "indexBounds" : {
                    "user_id" : [ 
                        "[\"00000000000040008000000000002a5d\", \"00000000000040008000000000002a5d\"]"
                    ],
                    "trk.0.direction" : [ 
                        "[\"ingress\", \"ingress\"]"
                    ],
                    "is_read" : [ 
                        "[MinKey, MaxKey]"
                    ],
                    "trk.0.data.status" : [ 
                        "[\"p_received\", \"p_received\"]", 
                        "[\"success\", \"success\"]"
                    ],
                    "is_removed" : [ 
                        "[false, false]"
                    ]
                }
            }
        },
        "rejectedPlans" : [ 
            {
                "stage" : "FETCH",
                "filter" : {
                    "$and" : [ 
                        {
                            "$or" : [ 
                                {
                                    "trk.0.extra_data.spam.is_spam" : {
                                        "$eq" : false
                                    }
                                }, 
                                {
                                    "$nor" : [ 
                                        {
                                            "trk.0.extra_data.spam.is_spam" : {
                                                "$exists" : true
                                            }
                                        }
                                    ]
                                }
                            ]
                        }, 
                        {
                            "is_removed" : {
                                "$eq" : false
                            }
                        }, 
                        {
                            "trk.0.direction" : {
                                "$eq" : "ingress"
                            }
                        }, 
                        {
                            "trk.0.type" : {
                                "$eq" : "fax"
                            }
                        }, 
                        {
                            "trk.0.data.status" : {
                                "$in" : [ 
                                    "p_received", 
                                    "success"
                                ]
                            }
                        }
                    ]
                },
                "inputStage" : {
                    "stage" : "IXSCAN",
                    "keyPattern" : {
                        "user_id" : 1,
                        "date" : -1
                    },
                    "indexName" : "user_id_date",
                    "isMultiKey" : false,
                    "isUnique" : false,
                    "isSparse" : false,
                    "isPartial" : false,
                    "indexVersion" : 1,
                    "direction" : "forward",
                    "indexBounds" : {
                        "user_id" : [ 
                            "[\"00000000000040008000000000002a5d\", \"00000000000040008000000000002a5d\"]"
                        ],
                        "date" : [ 
                            "(\"2018-01-09 10:36:31\", \"1970-01-01 00:00:00\")"
                        ]
                    }
                }
            }, 
            {
                "stage" : "FETCH",
                "filter" : {
                    "$and" : [ 
                        {
                            "$or" : [ 
                                {
                                    "trk.0.extra_data.spam.is_spam" : {
                                        "$eq" : false
                                    }
                                }, 
                                {
                                    "$nor" : [ 
                                        {
                                            "trk.0.extra_data.spam.is_spam" : {
                                                "$exists" : true
                                            }
                                        }
                                    ]
                                }
                            ]
                        }, 
                        {
                            "is_removed" : {
                                "$eq" : false
                            }
                        }, 
                        {
                            "trk.0.direction" : {
                                "$eq" : "ingress"
                            }
                        }, 
                        {
                            "trk.0.type" : {
                                "$eq" : "fax"
                            }
                        }, 
                        {
                            "date" : {
                                "$lt" : "2018-01-09 10:36:31"
                            }
                        }, 
                        {
                            "date" : {
                                "$gt" : "1970-01-01 00:00:00"
                            }
                        }, 
                        {
                            "trk.0.data.status" : {
                                "$in" : [ 
                                    "p_received", 
                                    "success"
                                ]
                            }
                        }
                    ]
                },
                "inputStage" : {
                    "stage" : "IXSCAN",
                    "keyPattern" : {
                        "user_id" : 1,
                        "to" : 1,
                        "from" : 1
                    },
                    "indexName" : "user_id_to_from",
                    "isMultiKey" : false,
                    "isUnique" : false,
                    "isSparse" : false,
                    "isPartial" : false,
                    "indexVersion" : 1,
                    "direction" : "forward",
                    "indexBounds" : {
                        "user_id" : [ 
                            "[\"00000000000040008000000000002a5d\", \"00000000000040008000000000002a5d\"]"
                        ],
                        "to" : [ 
                            "[MinKey, MaxKey]"
                        ],
                        "from" : [ 
                            "[MinKey, MaxKey]"
                        ]
                    }
                }
            }
        ]
    },
    "executionStats" : {
        "executionSuccess" : true,
        "nReturned" : 4682,
        "executionTimeMillis" : 2072,
        "totalKeysExamined" : 4688,
        "totalDocsExamined" : 4682,
        "executionStages" : {
            "stage" : "FETCH",
            "filter" : {
                "$and" : [ 
                    {
                        "$or" : [ 
                            {
                                "trk.0.extra_data.spam.is_spam" : {
                                    "$eq" : false
                                }
                            }, 
                            {
                                "$nor" : [ 
                                    {
                                        "trk.0.extra_data.spam.is_spam" : {
                                            "$exists" : true
                                        }
                                    }
                                ]
                            }
                        ]
                    }, 
                    {
                        "trk.0.type" : {
                            "$eq" : "fax"
                        }
                    }, 
                    {
                        "date" : {
                            "$lt" : "2018-01-09 10:36:31"
                        }
                    }, 
                    {
                        "date" : {
                            "$gt" : "1970-01-01 00:00:00"
                        }
                    }
                ]
            },
            "nReturned" : 4682,
            "executionTimeMillisEstimate" : 710,
            "works" : 4897,
            "advanced" : 4682,
            "needTime" : 5,
            "needYield" : 209,
            "saveState" : 234,
            "restoreState" : 234,
            "isEOF" : 1,
            "invalidates" : 1,
            "docsExamined" : 4682,
            "alreadyHasObj" : 0,
            "inputStage" : {
                "stage" : "IXSCAN",
                "nReturned" : 4682,
                "executionTimeMillisEstimate" : 305,
                "works" : 4688,
                "advanced" : 4682,
                "needTime" : 5,
                "needYield" : 0,
                "saveState" : 234,
                "restoreState" : 234,
                "isEOF" : 1,
                "invalidates" : 1,
                "keyPattern" : {
                    "user_id" : 1,
                    "trk.0.direction" : 1,
                    "is_read" : 1,
                    "trk.0.data.status" : 1,
                    "is_removed" : 1
                },
                "indexName" : "user_id_direction_is_read_status_is_removed",
                "isMultiKey" : false,
                "isUnique" : false,
                "isSparse" : false,
                "isPartial" : false,
                "indexVersion" : 1,
                "direction" : "forward",
                "indexBounds" : {
                    "user_id" : [ 
                        "[\"00000000000040008000000000002a5d\", \"00000000000040008000000000002a5d\"]"
                    ],
                    "trk.0.direction" : [ 
                        "[\"ingress\", \"ingress\"]"
                    ],
                    "is_read" : [ 
                        "[MinKey, MaxKey]"
                    ],
                    "trk.0.data.status" : [ 
                        "[\"p_received\", \"p_received\"]", 
                        "[\"success\", \"success\"]"
                    ],
                    "is_removed" : [ 
                        "[false, false]"
                    ]
                },
                "keysExamined" : 4688,
                "seeks" : 6,
                "dupsTested" : 0,
                "dupsDropped" : 0,
                "seenInvalidated" : 0
            }
        }
    },
    "serverInfo" : {
        "host" : “hs1.mydomain.com”,
        "port" : 27017,
        "version" : "3.4.10",
        "gitVersion" : "078f28920cb24de0dd479b5ea6c66c644f6326e9"
    },
    "ok" : 1.0
}

keysExamined просто 4688! Это не так много по сравнению с общими данными коллекции, которая составляет 30 миллионов документов. Когда Mongo замедляется, когда имеет эффект домино, загрузка процессора и памяти не высоки. Монго просто использует 40% памяти. Раздел диска Ext4 если это поможет

Еще один пример очень медленного запроса в деталях:

{
    "desc": "conn199276",
    "threadId": "140070259820288",
    "connectionId": 199276,
    "client": "client_server_ip:45590",
    "active": "true",
    "opid": 63869351,
    "secs_running": 36,
    "microsecs_running": 36136211,
    "op": "query",
    "ns": "cuda.call_history",
    "query": {
        "find": "call_history",
        "filter": {
            "is_removed": false,
            "trk.0.extra_data.spam.is_spam": true,
            "trk.0.direction": "ingress",
            "date": {
                "$gt": "1970-01-01 00:00:00",
                "$lt": "4001-01-01 00:00:00"
            },
            "trk.0.extra_data.status": {
                "$in": [
                    "success",
                    "p_received"
                ]
            },
            "trk.0.type": "clk",
            "owner_id": "00000000000040008000000000003828"
        },
        "sort": {
            "date": -1
        },
        "limit": 31
    },
    "numYields": 6600,
    "locks": {},
    "waitingForLock": "false",
    "lockStats": {
        "Global": {
            "acquireCount": {
                "r": 13200
            }
        },
        "MMAPV1Journal": {
            "acquireCount": {
                "r": 6611
            },
            "acquireWaitCount": {
                "r": 9
            },
            "timeAcquiringMicros": {
                "r": 50854
            }
        },
        "Database": {
            "acquireCount": {
                "r": 6600
            }
        },
        "Collection": {
            "acquireCount": {
                "R": 6600
            },
            "acquireWaitCount": {
                "R": 11
            },
            "timeAcquiringMicros": {
                "R": 163707
            }
        }
    }
}

Выход из db.stats():

rs0:PRIMARY> db.stats()
{
    "db" : "cuda",
    "collections" : 5,
    "views" : 0,
    "objects" : 55009248,
    "avgObjSize" : 2018.6135346551184,
    "dataSize" : 111042412544,
    "storageSize" : 113055362336,
    "numExtents" : 100,
    "indexes" : 7,
    "indexSize" : 14223460160,
    "fileSize" : 133012914176,
    "nsSizeMB" : 16,
    "extentFreeList" : {
        "num" : 0,
        "totalSize" : 0
    },
    "dataFileVersion" : {
        "major" : 4,
        "minor" : 22
    },
    "ok" : 1
}

mongostat отображает приведенный ниже результат, я думаю, номер ошибки высокий:

insert query update delete getmore command flushes mapped vsize   res faults qrw arw net_in net_out conn set repl                time
     5    93      4     *0       0    64|0       0         282G 9.11G     26 0|0 0|0  64.3k    187k  481 rs0  PRI Jan 10 06:25:14.476
    *0   107     *0      1       0    58|0       0         282G 9.14G      4 0|0 0|0  51.5k    247k  481 rs0  PRI Jan 10 06:25:15.475
     2    88      5     *0       0    70|0       0         282G 9.04G     26 0|0 0|0  61.5k    245k  481 rs0  PRI Jan 10 06:25:16.476
     3    98      2     *0       0    71|0       0         282G 9.12G      6 0|0 0|0  59.6k    274k  481 rs0  PRI Jan 10 06:25:17.474
     1   105     *0      1       0    82|0       0         282G 9.10G     14 0|0 0|0  63.4k    254k  481 rs0  PRI Jan 10 06:25:18.476
     1    88      2     *0       0    68|0       0         282G 9.12G     75 0|0 0|0  49.9k    194k  481 rs0  PRI Jan 10 06:25:19.476
     5   120      4     *0       0    78|0       0         282G 9.11G      5 0|0 0|0  75.7k    548k  484 rs0  PRI Jan 10 06:25:20.476
     3    89     *0      4       0    64|0       0         282G 9.14G     19 0|0 0|0  55.3k   1.59m  483 rs0  PRI Jan 10 06:25:21.476
    *0    81     *0     *0       0    62|0       0         282G 9.17G      0 0|0 0|0  46.9k    265k  481 rs0  PRI Jan 10 06:25:22.476
     6   105      3     *0       0    67|0       0         282G 9.15G     49 0|0 0|0  61.8k    240k  481 rs0  PRI Jan 10 06:25:23.475

И под нагрузкой у нас большие ошибки страницы:

insert query update delete getmore command flushes mapped vsize   res faults  qrw  arw net_in net_out conn set repl                time
     1    96     *0     *0       0    70|0       0         282G 10.9G    597  0|0  2|0  59.1k   1.06m  484 rs0  PRI Jan 10 07:48:45.061
     1   103      3     *0       0    79|0       0         282G 10.9G    715  0|0  0|0  66.4k    176k  484 rs0  PRI Jan 10 07:48:46.060
     2   102     *0     *0       0    72|0       0         282G 10.9G    718  0|0  2|0  59.6k    154k  484 rs0  PRI Jan 10 07:48:47.060
     2   104      2      1       0    83|0       0         282G 11.0G    692  0|0  3|0  68.7k    154k  483 rs0  PRI Jan 10 07:48:48.061
     2   103      2     *0       0    86|0       0         282G 11.0G    743  0|0  1|0  64.2k    154k  482 rs0  PRI Jan 10 07:48:49.059
    *0   116      2     *0       0    95|0       0         282G 11.0G    715  0|0  2|0  68.5k    237k  484 rs0  PRI Jan 10 07:48:50.060
     1   112      6      1       0    78|0       0         282G 11.0G    595  0|0  2|0  74.3k    231k  481 rs0  PRI Jan 10 07:48:51.061
     2   137      5      1       0    83|0       0         282G 11.0G    733  0|0  3|0  81.6k    301k  480 rs0  PRI Jan 10 07:48:52.060
     3   157      5      1       0    92|0       0         282G 11.0G    697  0|0  3|0  89.5k    183k  481 rs0  PRI Jan 10 07:48:53.059
     4   102      3     *0       0    73|0       0         282G 10.9G    599  0|0  1|0  67.1k    154k  480 rs0  PRI Jan 10 07:48:54.060

У нас есть около 15 ГБ свободной памяти, часть которой используется MongoDB для кеша!

На пике мы достигаем 400 OPS и 500 соединений в секунду. Что я могу сделать, чтобы улучшить производительность?

0 ответов

Другие вопросы по тегам