// #include "llviewerprecompiledheaders.h" #include "llfloatermessagelog.h" #include "lluictrlfactory.h" #include "llworld.h" #include "llviewerregion.h" #include "llscrolllistctrl.h" #include "lltexteditor.h" #include "llviewerwindow.h" // alertXml #include "llmessagetemplate.h" #include #include "llmenugl.h" #include "llfloatermessagebuilder.h" #include "llagent.h" //////////////////////////////// // LLFloaterMessageLogItem //////////////////////////////// #define MAX_PACKET_LEN (0x2000) LLTemplateMessageReader* LLFloaterMessageLogItem::sTemplateMessageReader = NULL; LLFloaterMessageLogItem::LLFloaterMessageLogItem(LLMessageLogEntry entry) : LLMessageLogEntry(entry.mType, entry.mFromHost, entry.mToHost, entry.mData, entry.mDataSize) { if(!sTemplateMessageReader) { sTemplateMessageReader = new LLTemplateMessageReader(gMessageSystem->mMessageNumbers); } mID.generate(); mSequenceID = 0; if(mType == TEMPLATE) { BOOL decode_invalid = FALSE; S32 decode_len = mDataSize; std::vector DecodeBuffer(MAX_PACKET_LEN,0); memcpy(&(DecodeBuffer[0]),&(mData[0]),decode_len); U8* decodep = &(DecodeBuffer[0]); mFlags = DecodeBuffer[0]; gMessageSystem->zeroCodeExpand(&decodep, &decode_len); if(decode_len < 7) decode_invalid = TRUE; else { mSequenceID = ntohl(*((U32*)(&decodep[1]))); sTemplateMessageReader->clearMessage(); if(!sTemplateMessageReader->validateMessage(decodep, decode_len, mFromHost, TRUE)) decode_invalid = TRUE; else { if(!sTemplateMessageReader->decodeData(decodep, mFromHost, TRUE)) decode_invalid = TRUE; else { LLMessageTemplate* temp = sTemplateMessageReader->getTemplate(); mName = temp->mName; mSummary = ""; if(mFlags) { mSummary.append(" [ "); if(mFlags & LL_ZERO_CODE_FLAG) mSummary.append(" Zer "); if(mFlags & LL_RELIABLE_FLAG) mSummary.append(" Rel "); if(mFlags & LL_RESENT_FLAG) mSummary.append(" Rsd "); if(mFlags & LL_ACK_FLAG) mSummary.append(" Ack "); mSummary.append(" ] "); } LLMessageTemplate::message_block_map_t::iterator blocks_end = temp->mMemberBlocks.end(); for (LLMessageTemplate::message_block_map_t::iterator blocks_iter = temp->mMemberBlocks.begin(); blocks_iter != blocks_end; ++blocks_iter) { LLMessageBlock* block = (*blocks_iter); const char* block_name = block->mName; S32 num_blocks = sTemplateMessageReader->getNumberOfBlocks(block_name); if(!num_blocks) mSummary.append(" { } "); else if(num_blocks > 1) mSummary.append(llformat(" %s [ %d ] { ... } ", block_name, num_blocks)); else for(S32 i = 0; i < 1; i++) { mSummary.append(" { "); LLMessageBlock::message_variable_map_t::iterator var_end = block->mMemberVariables.end(); for (LLMessageBlock::message_variable_map_t::iterator var_iter = block->mMemberVariables.begin(); var_iter != var_end; ++var_iter) { LLMessageVariable* variable = (*var_iter); const char* var_name = variable->getName(); BOOL returned_hex; std::string value = getString(sTemplateMessageReader, block_name, i, var_name, variable->getType(), returned_hex, TRUE); mSummary.append(llformat(" %s=%s ", var_name, value.c_str())); } mSummary.append(" } "); if(mSummary.length() > 255) break; } if(mSummary.length() > 255) { mSummary.append(" ... "); break; } } // blocks_iter } // decode_valid } } if(decode_invalid) { mName = "Invalid"; mSummary = ""; for(S32 i = 0; i < mDataSize; i++) mSummary.append(llformat("%02X ", mData[i])); } } else // not template { mName = "SOMETHING ELSE"; mSummary = "TODO: SOMETHING ELSE"; } } LLFloaterMessageLogItem::~LLFloaterMessageLogItem() { } BOOL LLFloaterMessageLogItem::isOutgoing() { return mFromHost == LLHost(16777343, gMessageSystem->getListenPort()); } std::string LLFloaterMessageLogItem::getFull(BOOL show_header) { std::string full(""); if(mType == TEMPLATE) { BOOL decode_invalid = FALSE; S32 decode_len = mDataSize; std::vector DecodeBuffer(MAX_PACKET_LEN,0); memcpy(&(DecodeBuffer[0]),&(mData[0]),decode_len); U8* decodep = &(DecodeBuffer[0]); gMessageSystem->zeroCodeExpand(&decodep, &decode_len); if(decode_len < 7) decode_invalid = TRUE; else { sTemplateMessageReader->clearMessage(); if(!sTemplateMessageReader->validateMessage(decodep, decode_len, mFromHost, TRUE)) decode_invalid = TRUE; else { if(!sTemplateMessageReader->decodeData(decodep, mFromHost, TRUE)) decode_invalid = TRUE; else { LLMessageTemplate* temp = sTemplateMessageReader->getTemplate(); full.append(isOutgoing() ? "out " : "in "); full.append(llformat("%s\n", temp->mName)); if(show_header) { full.append("[Header]\n"); full.append(llformat("SequenceID = %u\n", mSequenceID)); full.append(llformat("LL_ZERO_CODE_FLAG = %s\n", (mFlags & LL_ZERO_CODE_FLAG) ? "True" : "False")); full.append(llformat("LL_RELIABLE_FLAG = %s\n", (mFlags & LL_RELIABLE_FLAG) ? "True" : "False")); full.append(llformat("LL_RESENT_FLAG = %s\n", (mFlags & LL_RESENT_FLAG) ? "True" : "False")); full.append(llformat("LL_ACK_FLAG = %s\n", (mFlags & LL_ACK_FLAG) ? "True" : "False")); } LLMessageTemplate::message_block_map_t::iterator blocks_end = temp->mMemberBlocks.end(); for (LLMessageTemplate::message_block_map_t::iterator blocks_iter = temp->mMemberBlocks.begin(); blocks_iter != blocks_end; ++blocks_iter) { LLMessageBlock* block = (*blocks_iter); const char* block_name = block->mName; S32 num_blocks = sTemplateMessageReader->getNumberOfBlocks(block_name); for(S32 i = 0; i < num_blocks; i++) { full.append(llformat("[%s]\n", block->mName)); LLMessageBlock::message_variable_map_t::iterator var_end = block->mMemberVariables.end(); for (LLMessageBlock::message_variable_map_t::iterator var_iter = block->mMemberVariables.begin(); var_iter != var_end; ++var_iter) { LLMessageVariable* variable = (*var_iter); const char* var_name = variable->getName(); BOOL returned_hex; std::string value = getString(sTemplateMessageReader, block_name, i, var_name, variable->getType(), returned_hex); if(returned_hex) full.append(llformat("%s =| ", var_name)); else full.append(llformat("%s = ", var_name)); // llformat has a 1024 char limit!? full.append(value); full.append("\n"); } } } // blocks_iter } // decode_valid } } if(decode_invalid) { full = isOutgoing() ? "out" : "in"; full.append("\n"); for(S32 i = 0; i < mDataSize; i++) full.append(llformat("%02X ", mData[i])); } } else // not template { full = "FIXME"; } return full; } // static std::string LLFloaterMessageLogItem::getString(LLTemplateMessageReader* readerp, const char* block_name, S32 block_num, const char* var_name, e_message_variable_type var_type, BOOL &returned_hex, BOOL summary_mode) { returned_hex = FALSE; std::stringstream stream; char* value; U32 valueU32; U16 valueU16; LLVector3 valueVector3; LLVector3d valueVector3d; LLVector4 valueVector4; LLQuaternion valueQuaternion; LLUUID valueLLUUID; switch(var_type) { case MVT_U8: U8 valueU8; readerp->getU8(block_name, var_name, valueU8, block_num); stream << U32(valueU8); break; case MVT_U16: readerp->getU16(block_name, var_name, valueU16, block_num); stream << valueU16; break; case MVT_U32: readerp->getU32(block_name, var_name, valueU32, block_num); stream << valueU32; break; case MVT_U64: U64 valueU64; readerp->getU64(block_name, var_name, valueU64, block_num); stream << valueU64; break; case MVT_S8: S8 valueS8; readerp->getS8(block_name, var_name, valueS8, block_num); stream << S32(valueS8); break; case MVT_S16: S16 valueS16; readerp->getS16(block_name, var_name, valueS16, block_num); stream << valueS16; break; case MVT_S32: S32 valueS32; readerp->getS32(block_name, var_name, valueS32, block_num); stream << valueS32; break; /*case MVT_S64: S64 valueS64; readerp->getS64(block_name, var_name, valueS64, block_num); stream << valueS64; break;*/ case MVT_F32: F32 valueF32; readerp->getF32(block_name, var_name, valueF32, block_num); stream << valueF32; break; case MVT_F64: F64 valueF64; readerp->getF64(block_name, var_name, valueF64, block_num); stream << valueF64; break; case MVT_LLVector3: readerp->getVector3(block_name, var_name, valueVector3, block_num); //stream << valueVector3; stream << "<" << valueVector3.mV[0] << ", " << valueVector3.mV[1] << ", " << valueVector3.mV[2] << ">"; break; case MVT_LLVector3d: readerp->getVector3d(block_name, var_name, valueVector3d, block_num); //stream << valueVector3d; stream << "<" << valueVector3d.mdV[0] << ", " << valueVector3d.mdV[1] << ", " << valueVector3d.mdV[2] << ">"; break; case MVT_LLVector4: readerp->getVector4(block_name, var_name, valueVector4, block_num); //stream << valueVector4; stream << "<" << valueVector4.mV[0] << ", " << valueVector4.mV[1] << ", " << valueVector4.mV[2] << ", " << valueVector4.mV[3] << ">"; break; case MVT_LLQuaternion: readerp->getQuat(block_name, var_name, valueQuaternion, block_num); //stream << valueQuaternion; stream << "<" << valueQuaternion.mQ[0] << ", " << valueQuaternion.mQ[1] << ", " << valueQuaternion.mQ[2] << ", " << valueQuaternion.mQ[3] << ">"; break; case MVT_LLUUID: readerp->getUUID(block_name, var_name, valueLLUUID, block_num); stream << valueLLUUID; break; case MVT_BOOL: BOOL valueBOOL; readerp->getBOOL(block_name, var_name, valueBOOL, block_num); stream << valueBOOL; break; case MVT_IP_ADDR: readerp->getIPAddr(block_name, var_name, valueU32, block_num); stream << LLHost(valueU32, 0).getIPString(); break; case MVT_IP_PORT: readerp->getIPPort(block_name, var_name, valueU16, block_num); stream << valueU16; case MVT_VARIABLE: case MVT_FIXED: default: S32 size = readerp->getSize(block_name, block_num, var_name); if(size) { value = new char[size + 1]; readerp->getBinaryData(block_name, var_name, value, size, block_num); value[size] = '\0'; S32 readable = 0; S32 unreadable = 0; S32 end = (summary_mode && (size > 64)) ? 64 : size; for(S32 i = 0; i < end; i++) { if(!value[i]) { if(i != (end - 1)) { // don't want null terminator hiding data unreadable = S32_MAX; break; } } else if(value[i] < 0x20 || value[i] >= 0x7F) { if(summary_mode) unreadable++; else { // never want any wrong characters outside of summary mode unreadable = S32_MAX; break; } } else readable++; } if(readable >= unreadable) { if(summary_mode && (size > 64)) { for(S32 i = 60; i < 63; i++) value[i] = '.'; value[63] = '\0'; } stream << value; delete[] value; } else { returned_hex = TRUE; S32 end = (summary_mode && (size > 8)) ? 8 : size; for(S32 i = 0; i < end; i++) //stream << std::uppercase << std::hex << U32(value[i]) << " "; stream << llformat("%02X ", (U8)value[i]); if(summary_mode && (size > 8)) stream << " ... "; } } break; } return stream.str(); } LLMessageLogFilter::LLMessageLogFilter() { } LLMessageLogFilter::~LLMessageLogFilter() { } BOOL LLMessageLogFilter::set(std::string filter) { mPositiveNames.clear(); mNegativeNames.clear(); typedef boost::tokenizer > tokenizer; boost::char_separator sep(" ","",boost::keep_empty_tokens); boost::tokenizer > tokens(filter, sep); boost::tokenizer >::iterator end = tokens.end(); for(boost::tokenizer >::iterator iter = tokens.begin(); iter != end; ++iter) { std::string token = (*iter); LLStringUtil::trim(token); LLStringUtil::toLower(token); BOOL negative = token.find("!") == 0; if(negative) { token = token.substr(1); mNegativeNames.push_back(token); } else mPositiveNames.push_back(token); } return TRUE; } //////////////////////////////// // LLMessageLogFilterApply //////////////////////////////// LLMessageLogFilterApply::LLMessageLogFilterApply() : LLEventTimer(0.1f), mFinished(FALSE), mProgress(0) { mIter = LLFloaterMessageLog::sMessageLogEntries.begin(); } void LLMessageLogFilterApply::cancel() { mFinished = TRUE; } BOOL LLMessageLogFilterApply::tick() { std::deque::iterator end = LLFloaterMessageLog::sMessageLogEntries.end(); if(mIter == end || !LLFloaterMessageLog::sInstance) { mFinished = TRUE; if(LLFloaterMessageLog::sInstance) { if(LLFloaterMessageLog::sInstance->mMessageLogFilterApply == this) { LLFloaterMessageLog::sInstance->stopApplyingFilter(); } } return TRUE; } for(S32 i = 0; i < 256; i++) { if(mIter == end) { mFinished = TRUE; if(LLFloaterMessageLog::sInstance) { if(LLFloaterMessageLog::sInstance->mMessageLogFilterApply == this) { LLFloaterMessageLog::sInstance->stopApplyingFilter(); //we're done messing with the deque, push all queued items to the main deque std::deque::iterator queueIter = mQueuedMessages.begin(); std::deque::iterator queueEnd = mQueuedMessages.end(); while(queueIter != queueEnd) { LLFloaterMessageLog::sInstance->conditionalLog(LLFloaterMessageLogItem((*queueIter))); ++queueIter; } mQueuedMessages.clear(); } } return TRUE; } LLFloaterMessageLog::sInstance->conditionalLog(LLFloaterMessageLogItem((*mIter))); mIter++; mProgress++; } LLFloaterMessageLog::sInstance->updateFilterStatus(); return FALSE; } //////////////////////////////// // LLFloaterMessageLog //////////////////////////////// LLFloaterMessageLog* LLFloaterMessageLog::sInstance; std::list LLFloaterMessageLog::sNetListItems; std::deque LLFloaterMessageLog::sMessageLogEntries; std::vector LLFloaterMessageLog::sFloaterMessageLogItems; LLMessageLogFilter LLFloaterMessageLog::sMessageLogFilter = LLMessageLogFilter(); std::string LLFloaterMessageLog::sMessageLogFilterString("!StartPingCheck !CompletePingCheck !PacketAck !SimulatorViewerTimeMessage !SimStats !AgentUpdate !AgentAnimation !AvatarAnimation !ViewerEffect !CoarseLocationUpdate !LayerData !CameraConstraint !ObjectUpdateCached !RequestMultipleObjects !ObjectUpdate !ObjectUpdateCompressed !ImprovedTerseObjectUpdate !KillObject !ImagePacket !SendXferPacket !ConfirmXferPacket !TransferPacket !SoundTrigger !AttachedSound !PreloadSound"); BOOL LLFloaterMessageLog::sBusyApplyingFilter = FALSE; LLFloaterMessageLog::LLFloaterMessageLog() : LLFloater(), LLEventTimer(1.0f), mNetInfoMode(NI_NET), mMessageLogFilterApply(NULL) { sInstance = this; LLMessageLog::setCallback(onLog); sMessageLogEntries = LLMessageLog::getDeque(); LLUICtrlFactory::getInstance()->buildFloater(this, "floater_message_log.xml"); } LLFloaterMessageLog::~LLFloaterMessageLog() { LLMessageLog::setCallback(NULL); stopApplyingFilter(); sInstance = NULL; sNetListItems.clear(); sMessageLogEntries.clear(); sFloaterMessageLogItems.clear(); } // static void LLFloaterMessageLog::show() { if(!sInstance) sInstance = new LLFloaterMessageLog(); sInstance->open(); } BOOL LLFloaterMessageLog::postBuild() { childSetCommitCallback("net_list", onCommitNetList, this); childSetCommitCallback("message_log", onCommitMessageLog, this); childSetAction("filter_choice_btn", onClickFilterChoice, this); childSetAction("filter_apply_btn", onClickFilterApply, this); childSetCommitCallback("filter_edit", onCommitFilter, this); childSetAction("clear_log_btn", onClickClearLog, this); childSetAction("send_to_message_builder_btn", onClickSendToMessageBuilder, this); childSetText("filter_edit", sMessageLogFilterString); refreshNetList(); refreshNetInfo(TRUE); startApplyingFilter(sMessageLogFilterString, TRUE); return TRUE; } BOOL LLFloaterMessageLog::tick() { refreshNetList(); refreshNetInfo(FALSE); return FALSE; } LLNetListItem* LLFloaterMessageLog::findNetListItem(LLHost host) { std::list::iterator end = sNetListItems.end(); for(std::list::iterator iter = sNetListItems.begin(); iter != end; ++iter) if((*iter)->mCircuitData && (*iter)->mCircuitData->getHost() == host) return (*iter); return NULL; } LLNetListItem* LLFloaterMessageLog::findNetListItem(LLUUID id) { std::list::iterator end = sNetListItems.end(); for(std::list::iterator iter = sNetListItems.begin(); iter != end; ++iter) if((*iter)->mID == id) return (*iter); return NULL; } void LLFloaterMessageLog::refreshNetList() { LLScrollListCtrl* scrollp = getChild("net_list"); // Update circuit data of net list items std::vector circuits = gMessageSystem->getCircuit()->getCircuitDataList(); std::vector::iterator circuits_end = circuits.end(); for(std::vector::iterator iter = circuits.begin(); iter != circuits_end; ++iter) { LLNetListItem* itemp = findNetListItem((*iter)->getHost()); if(!itemp) { LLUUID id; id.generate(); itemp = new LLNetListItem(id); sNetListItems.push_back(itemp); } itemp->mCircuitData = (*iter); } // Clear circuit data of items whose circuits are gone std::list::iterator items_end = sNetListItems.end(); for(std::list::iterator iter = sNetListItems.begin(); iter != items_end; ++iter) { if(std::find(circuits.begin(), circuits.end(), (*iter)->mCircuitData) == circuits.end()) (*iter)->mCircuitData = NULL; } // Remove net list items that are totally useless now for(std::list::iterator iter = sNetListItems.begin(); iter != sNetListItems.end();) { if((*iter)->mCircuitData == NULL) iter = sNetListItems.erase(iter); else ++iter; } // Update names of net list items items_end = sNetListItems.end(); for(std::list::iterator iter = sNetListItems.begin(); iter != items_end; ++iter) { LLNetListItem* itemp = (*iter); if(itemp->mAutoName) { if(itemp->mCircuitData) { LLViewerRegion* regionp = LLWorld::getInstance()->getRegion(itemp->mCircuitData->getHost()); if(regionp) { std::string name = regionp->getName(); if(name == "") name = llformat("%s (awaiting region name)", itemp->mCircuitData->getHost().getString().c_str()); itemp->mName = name; itemp->mPreviousRegionName = name; } else { itemp->mName = itemp->mCircuitData->getHost().getString(); if(itemp->mPreviousRegionName != "") itemp->mName.append(llformat(" (was %s)", itemp->mPreviousRegionName.c_str())); } } else { // an item just for an event queue, not handled yet itemp->mName = "Something else"; } } } // Rebuild scroll list from scratch LLUUID selected_id = scrollp->getFirstSelected() ? scrollp->getFirstSelected()->getUUID() : LLUUID::null; S32 scroll_pos = scrollp->getScrollPos(); scrollp->clearRows(); for(std::list::iterator iter = sNetListItems.begin(); iter != items_end; ++iter) { LLNetListItem* itemp = (*iter); LLSD element; element["id"] = itemp->mID; LLSD& text_column = element["columns"][0]; text_column["column"] = "text"; text_column["value"] = itemp->mName + (itemp->mCircuitData->getHost() == gAgent.getRegionHost() ? " (main)" : ""); for(int i = 0; i < 2; i++) { LLSD& icon_column = element["columns"][i + 1]; icon_column["column"] = llformat("icon%d", i); icon_column["type"] = "icon"; icon_column["value"] = ""; } LLScrollListItem* scroll_itemp = scrollp->addElement(element); BOOL has_live_circuit = itemp->mCircuitData && itemp->mCircuitData->isAlive(); if(has_live_circuit) { LLScrollListIcon* icon = (LLScrollListIcon*)scroll_itemp->getColumn(1); icon->setValue("icon_net_close_circuit.tga"); icon->setClickCallback(onClickCloseCircuit, itemp); } else { LLScrollListIcon* icon = (LLScrollListIcon*)scroll_itemp->getColumn(1); icon->setValue("icon_net_close_circuit_gray.tga"); icon->setClickCallback(NULL, NULL); } // Event queue isn't even supported yet... FIXME LLScrollListIcon* icon = (LLScrollListIcon*)scroll_itemp->getColumn(2); icon->setValue("icon_net_close_eventpoll_gray.tga"); icon->setClickCallback(NULL, NULL); } if(selected_id.notNull()) scrollp->selectByID(selected_id); if(scroll_pos < scrollp->getItemCount()) scrollp->setScrollPos(scroll_pos); } void LLFloaterMessageLog::refreshNetInfo(BOOL force) { if(mNetInfoMode != NI_NET) return; LLScrollListCtrl* scrollp = getChild("net_list"); LLScrollListItem* selected_itemp = scrollp->getFirstSelected(); if(selected_itemp) { if(!force) if(getChild("net_info")->hasSelection()) return; LLNetListItem* itemp = findNetListItem(selected_itemp->getUUID()); if(itemp) { std::string info(llformat("%s\n--------------------------------\n\n", itemp->mName.c_str())); if(itemp->mCircuitData) { LLCircuitData* cdp = itemp->mCircuitData; info.append("Circuit\n--------------------------------\n"); info.append(llformat(" * Host: %s\n", cdp->getHost().getString().c_str())); S32 seconds = (S32)cdp->getAgeInSeconds(); S32 minutes = seconds / 60; seconds = seconds % 60; S32 hours = minutes / 60; minutes = minutes % 60; info.append(llformat(" * Age: %dh %dm %ds\n", hours, minutes, seconds)); info.append(llformat(" * Alive: %s\n", cdp->isAlive() ? "yes" : "no")); info.append(llformat(" * Blocked: %s\n", cdp->isBlocked() ? "yes" : "no")); info.append(llformat(" * Allow timeout: %s\n", cdp->getAllowTimeout() ? "yes" : "no")); info.append(llformat(" * Trusted: %s\n", cdp->getTrusted() ? "yes" : "no")); info.append(llformat(" * Ping delay: %d\n", cdp->getPingDelay())); info.append(llformat(" * Packets out: %d\n", cdp->getPacketsOut())); info.append(llformat(" * Bytes out: %d\n", cdp->getBytesOut())); info.append(llformat(" * Packets in: %d\n", cdp->getPacketsIn())); info.append(llformat(" * Bytes in: %d\n", cdp->getBytesIn())); info.append(llformat(" * Endpoint ID: %s\n", cdp->getLocalEndPointID().asString().c_str())); info.append(llformat(" * Remote ID: %s\n", cdp->getRemoteID().asString().c_str())); info.append(llformat(" * Remote session ID: %s\n", cdp->getRemoteSessionID().asString().c_str())); } childSetText("net_info", info); } else childSetText("net_info", std::string("")); } else childSetText("net_info", std::string("")); } void LLFloaterMessageLog::setNetInfoMode(ENetInfoMode mode) { mNetInfoMode = mode; if(mNetInfoMode == NI_NET) refreshNetInfo(TRUE); childSetEnabled("send_to_message_builder_btn", mNetInfoMode == NI_LOG); } // static void LLFloaterMessageLog::onLog(LLMessageLogEntry entry) { //don't mess with the queue while a filter's being applied, or face invalid iterators if(!sBusyApplyingFilter) { sMessageLogEntries.push_back(entry); conditionalLog(LLFloaterMessageLogItem(entry)); } } // static void LLFloaterMessageLog::conditionalLog(LLFloaterMessageLogItem item) { if(!sBusyApplyingFilter) sInstance->childSetText("log_status_text", llformat("Showing %d messages from %d", sFloaterMessageLogItems.size(), sMessageLogEntries.size())); std::string find_name = item.mName; LLStringUtil::toLower(find_name); if(sMessageLogFilter.mPositiveNames.size()) if(std::find(sMessageLogFilter.mPositiveNames.begin(), sMessageLogFilter.mPositiveNames.end(), find_name) == sMessageLogFilter.mPositiveNames.end()) return; if(std::find(sMessageLogFilter.mNegativeNames.begin(), sMessageLogFilter.mNegativeNames.end(), find_name) != sMessageLogFilter.mNegativeNames.end()) return; sFloaterMessageLogItems.push_back(item); // moved from beginning... BOOL outgoing = item.isOutgoing(); std::string net_name("\?\?\?"); if(item.mType == LLFloaterMessageLogItem::TEMPLATE) { LLHost find_host = outgoing ? item.mToHost : item.mFromHost; net_name = find_host.getIPandPort(); std::list::iterator end = sNetListItems.end(); for(std::list::iterator iter = sNetListItems.begin(); iter != end; ++iter) { if((*iter)->mCircuitData->getHost() == find_host) { net_name = (*iter)->mName; break; } } } LLSD element; element["id"] = item.mID; LLSD& sequence_column = element["columns"][0]; sequence_column["column"] = "sequence"; sequence_column["value"] = llformat("%u", item.mSequenceID); LLSD& type_column = element["columns"][1]; type_column["column"] = "type"; type_column["value"] = item.mType == LLFloaterMessageLogItem::TEMPLATE ? "UDP" : "\?\?\?"; LLSD& direction_column = element["columns"][2]; direction_column["column"] = "direction"; direction_column["value"] = outgoing ? "to" : "from"; LLSD& net_column = element["columns"][3]; net_column["column"] = "net"; net_column["value"] = net_name; LLSD& name_column = element["columns"][4]; name_column["column"] = "name"; name_column["value"] = item.mName; /* LLSD& zer_column = element["columns"][5]; zer_column["column"] = "flag_zer"; zer_column["type"] = "icon"; zer_column["value"] = (item.mFlags & LL_ZERO_CODE_FLAG) ? "flag_zer.tga" : ""; LLSD& rel_column = element["columns"][6]; rel_column["column"] = "flag_rel"; rel_column["type"] = "icon"; rel_column["value"] = (item.mFlags & LL_RELIABLE_FLAG) ? "flag_rel.tga" : ""; LLSD& rsd_column = element["columns"][7]; rsd_column["column"] = "flag_rsd"; rsd_column["type"] = "icon"; rsd_column["value"] = (item.mFlags & LL_RESENT_FLAG) ? "flag_rsd.tga" : ""; LLSD& ack_column = element["columns"][8]; ack_column["column"] = "flag_ack"; ack_column["type"] = "icon"; ack_column["value"] = (item.mFlags & LL_ACK_FLAG) ? "flag_ack.tga" : ""; */ LLSD& summary_column = element["columns"][5]; summary_column["column"] = "summary"; summary_column["value"] = item.mSummary; LLScrollListCtrl* scrollp = sInstance->getChild("message_log"); S32 scroll_pos = scrollp->getScrollPos(); scrollp->addElement(element); if(scroll_pos > scrollp->getItemCount() - scrollp->getPageLines() - 4) scrollp->setScrollPos(scrollp->getItemCount()); } // static void LLFloaterMessageLog::onCommitNetList(LLUICtrl* ctrl, void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; floaterp->setNetInfoMode(NI_NET); floaterp->refreshNetInfo(TRUE); } // static void LLFloaterMessageLog::onCommitMessageLog(LLUICtrl* ctrl, void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; LLScrollListCtrl* scrollp = floaterp->getChild("message_log"); LLScrollListItem* selected_itemp = scrollp->getFirstSelected(); if(!selected_itemp) return; LLUUID id = selected_itemp->getUUID(); std::vector::iterator end = sFloaterMessageLogItems.end(); for(std::vector::iterator iter = sFloaterMessageLogItems.begin(); iter != end; ++iter) { if(iter->mID == id) { floaterp->setNetInfoMode(NI_LOG); floaterp->childSetText("net_info", iter->getFull(FALSE)); break; } } } // static BOOL LLFloaterMessageLog::onClickCloseCircuit(void* user_data) { LLNetListItem* itemp = (LLNetListItem*)user_data; LLCircuitData* cdp = (LLCircuitData*)itemp->mCircuitData; if(!cdp) return FALSE; LLHost myhost = cdp->getHost(); LLSD args; args["MESSAGE"] = "This will delete local circuit data.\nDo you want to tell the remote host to close the circuit too?"; LLSD payload; payload["circuittoclose"] = myhost.getString(); LLNotifications::instance().add("GenericAlertYesCancel", args, payload, onConfirmCloseCircuit); return TRUE; } // static bool LLFloaterMessageLog::onConfirmCloseCircuit(const LLSD& notification, const LLSD& response ) { S32 option = LLNotification::getSelectedOption(notification, response); LLCircuitData* cdp = gMessageSystem->mCircuitInfo.findCircuit(LLHost(notification["payload"]["circuittoclose"].asString())); if(!cdp) return false; LLViewerRegion* regionp = LLWorld::getInstance()->getRegion(cdp->getHost()); switch(option) { case 0: // yes gMessageSystem->newMessageFast(_PREHASH_CloseCircuit); gMessageSystem->sendReliable(cdp->getHost()); break; case 2: // cancel return false; break; case 1: // no default: break; } if(gMessageSystem->findCircuitCode(cdp->getHost())) gMessageSystem->disableCircuit(cdp->getHost()); else gMessageSystem->getCircuit()->removeCircuitData(cdp->getHost()); if(regionp) { LLHost myhost = regionp->getHost(); LLSD args; args["MESSAGE"] = "That host had a region associated with it.\nDo you want to clean that up?"; LLSD payload; payload["regionhost"] = myhost.getString(); LLNotifications::instance().add("GenericAlertYesCancel", args, payload, onConfirmRemoveRegion); } return false; } // static bool LLFloaterMessageLog::onConfirmRemoveRegion(const LLSD& notification, const LLSD& response ) { S32 option = LLNotification::getSelectedOption(notification, response); if(option == 0) // yes LLWorld::getInstance()->removeRegion(LLHost(notification["payload"]["regionhost"].asString())); return false; } // static void LLFloaterMessageLog::onClickFilterApply(void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; floaterp->startApplyingFilter(floaterp->childGetValue("filter_edit"), FALSE); } void LLFloaterMessageLog::startApplyingFilter(std::string filter, BOOL force) { LLMessageLogFilter new_filter = LLMessageLogFilter(); sMessageLogFilterString = filter; new_filter.set(sMessageLogFilterString); if(!filter.length() || filter.at(filter.length()-1) != ' ') childSetText("filter_edit", filter + " "); if(force || (new_filter.mNegativeNames != sMessageLogFilter.mNegativeNames) || (new_filter.mPositiveNames != sMessageLogFilter.mPositiveNames)) { stopApplyingFilter(); sMessageLogFilter = new_filter; sFloaterMessageLogItems.clear(); getChild("message_log")->clearRows(); sBusyApplyingFilter = TRUE; childSetVisible("message_log", false); //childSetVisible("log_status_text", true); mMessageLogFilterApply = new LLMessageLogFilterApply(); } } void LLFloaterMessageLog::stopApplyingFilter() { if(mMessageLogFilterApply) { if(!(mMessageLogFilterApply->mFinished)) mMessageLogFilterApply->cancel(); //delete mMessageLogFilterApply; sBusyApplyingFilter = FALSE; //childSetVisible("log_status_text", false); childSetVisible("message_log", true); childSetText("log_status_text", llformat("Showing %d messages from %d", sFloaterMessageLogItems.size(), sMessageLogEntries.size())); } } void LLFloaterMessageLog::updateFilterStatus() { if(!mMessageLogFilterApply || !sBusyApplyingFilter) return; S32 progress = mMessageLogFilterApply->mProgress; S32 packets = sMessageLogEntries.size(); S32 matches = sFloaterMessageLogItems.size(); std::string text = llformat("Applying filter ( %d / %d ), %d matches ...", progress, packets, matches); childSetText("log_status_text", text); } // static void LLFloaterMessageLog::onCommitFilter(LLUICtrl* ctrl, void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; floaterp->startApplyingFilter(floaterp->childGetValue("filter_edit"), FALSE); } // static void LLFloaterMessageLog::onClickClearLog(void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; floaterp->stopApplyingFilter(); floaterp->getChild("message_log")->clearRows(); floaterp->setNetInfoMode(NI_NET); sMessageLogEntries.clear(); sFloaterMessageLogItems.clear(); } // static void LLFloaterMessageLog::onClickFilterChoice(void* user_data) { LLMenuGL* menu = new LLMenuGL(LLStringUtil::null); menu->append(new LLMenuItemCallGL("No filter", onClickFilterMenu, NULL, (void*)"")); menu->append(new LLMenuItemCallGL("Fewer spammy messages", onClickFilterMenu, NULL, (void*)"!StartPingCheck !CompletePingCheck !PacketAck !SimulatorViewerTimeMessage !SimStats !AgentUpdate !AgentAnimation !AvatarAnimation !ViewerEffect !CoarseLocationUpdate !LayerData !CameraConstraint !ObjectUpdateCached !RequestMultipleObjects !ObjectUpdate !ObjectUpdateCompressed !ImprovedTerseObjectUpdate !KillObject !ImagePacket !SendXferPacket !ConfirmXferPacket !TransferPacket")); menu->append(new LLMenuItemCallGL("Fewer spammy messages (minus sound crap)", onClickFilterMenu, NULL, (void*)"!StartPingCheck !CompletePingCheck !PacketAck !SimulatorViewerTimeMessage !SimStats !AgentUpdate !AgentAnimation !AvatarAnimation !ViewerEffect !CoarseLocationUpdate !LayerData !CameraConstraint !ObjectUpdateCached !RequestMultipleObjects !ObjectUpdate !ObjectUpdateCompressed !ImprovedTerseObjectUpdate !KillObject !ImagePacket !SendXferPacket !ConfirmXferPacket !TransferPacket !SoundTrigger !AttachedSound !PreloadSound")); menu->append(new LLMenuItemCallGL("Object updates", onClickFilterMenu, NULL, (void*)"ObjectUpdateCached ObjectUpdate ObjectUpdateCompressed ImprovedTerseObjectUpdate KillObject RequestMultipleObjects")); menu->append(new LLMenuItemCallGL("Abnormal", onClickFilterMenu, NULL, (void*)"Invalid TestMessage AddCircuitCode NeighborList AvatarTextureUpdate SimulatorMapUpdate SimulatorSetMap SubscribeLoad UnsubscribeLoad SimulatorReady SimulatorPresentAtLocation SimulatorLoad SimulatorShutdownRequest RegionPresenceRequestByRegionID RegionPresenceRequestByHandle RegionPresenceResponse UpdateSimulator LogDwellTime FeatureDisabled LogFailedMoneyTransaction UserReportInternal SetSimStatusInDatabase SetSimPresenceInDatabase OpenCircuit CloseCircuit DirFindQueryBackend DirPlacesQueryBackend DirClassifiedQueryBackend DirLandQueryBackend DirPopularQueryBackend GroupNoticeAdd DataHomeLocationRequest DataHomeLocationReply DerezContainer ObjectCategory ObjectExportSelected StateSave ReportAutosaveCrash AgentAlertMessage NearestLandingRegionRequest NearestLandingRegionReply NearestLandingRegionUpdated TeleportLandingStatusChanged ConfirmEnableSimulator KickUserAck SystemKickUser AvatarPropertiesRequestBackend UpdateParcel RemoveParcel MergeParcel LogParcelChanges CheckParcelSales ParcelSales StartAuction ConfirmAuctionStart CompleteAuction CancelAuction CheckParcelAuctions ParcelAuctions ChatPass EdgeDataPacket SimStatus ChildAgentUpdate ChildAgentAlive ChildAgentPositionUpdate ChildAgentDying ChildAgentUnknown AtomicPassObject KillChildAgents ScriptSensorRequest ScriptSensorReply DataServerLogout RequestInventoryAsset InventoryAssetResponse TransferInventory TransferInventoryAck EventLocationRequest EventLocationReply MoneyTransferBackend RoutedMoneyBalanceReply SetStartLocation NetTest SetCPURatio SimCrashed NameValuePair RemoveNameValuePair UpdateAttachment RemoveAttachment EmailMessageRequest EmailMessageReply InternalScriptMail ScriptDataRequest ScriptDataReply InviteGroupResponse TallyVotes LiveHelpGroupRequest LiveHelpGroupReply GroupDataUpdate LogTextMessage CreateTrustedCircuit ParcelRename SystemMessage RpcChannelRequest RpcChannelReply RpcScriptRequestInbound RpcScriptRequestInboundForward RpcScriptReplyInbound ScriptMailRegistration Error")); menu->updateParent(LLMenuGL::sMenuContainer); menu->setCanTearOff(FALSE); LLView* buttonp = sInstance->getChild("filter_choice_btn"); S32 x = buttonp->getRect().mLeft; S32 y = buttonp->getRect().mBottom; LLMenuGL::showPopup(sInstance, menu, x, y); } // static void LLFloaterMessageLog::onClickFilterMenu(void* user_data) { std::string filter = std::string((char*)user_data); sInstance->childSetText("filter_edit", filter); sInstance->startApplyingFilter(filter, FALSE); } // static void LLFloaterMessageLog::onClickSendToMessageBuilder(void* user_data) { LLFloaterMessageLog* floaterp = (LLFloaterMessageLog*)user_data; LLScrollListCtrl* scrollp = floaterp->getChild("message_log"); LLScrollListItem* selected_itemp = scrollp->getFirstSelected(); if(!selected_itemp) return; LLUUID id = selected_itemp->getUUID(); std::vector::iterator end = sFloaterMessageLogItems.end(); for(std::vector::iterator iter = sFloaterMessageLogItems.begin(); iter != end; ++iter) { if(iter->mID == id) { std::string message_text = iter->getFull(FALSE); LLFloaterMessageBuilder::show(message_text); break; } } } //