2014-02-26 78 views
4

我使用基于SolrJ的客户端来查询Solr,并且我一直试图构建HTTP请求,其中排除了facet名称/值对。我正在使用的Web界面有一个改进的进一步功能,允许排除一个或多个方面值。我有3个方面的领域:域名,内容类型和作者,我希望能够通过排除他们每个人来处理faceting。例如,q = Dickensfq=-author:Dickens, Janet将构建以下HTTP请求:使用SolrJ过滤器查询排除

/solr/solrbase/select?q=Dickens&fq=-author:Dickens%2c+Janet&wt=json&indent=true 

尽管XML转储将是这样的:

   <facets> 
       <facet name="author"> 
       <facetEntry count="20">Dickens, Charles</facetEntry> 
       <facetEntry count="10">Dickens, Sarah</facetEntry> 
       </facet> 
      </facets> 

到目前为止,Java实现,我有工作似乎并不处理筛选查询排除:

private HttpSolrServer solrServer; 
solrServer = new HttpSolrServer("http://localhost:8983/solr/"); 

private static final String CONFIG_SOLR_FACET_FIELD = "facet_field"; 
private String[] _facetFields = new String[] {"author"}; 

private static final String CONFIG_SOLR_FACETS = "facets" 
    Element el = myParams.getChild(CONFIG_SOLR_FACETS); 

     _facetUse = el.getAttributeValue("useFacets", "true"); 
     _facetMinCount = el.getAttributeValue("minCount", String.valueOf(1)); 
     _facetLimit = el.getAttributeValue("limit", String.valueOf(20)); 


List vals = el.getChildren(CONFIG_SOLR_FACET_FIELD); 
     if (vals.size() > 0) { 
      _facetFields = new String[vals.size()]; 
      for (int i=0; i < vals.size(); i++) { 
      _facetFields[i] = ((Element)vals.get(i)).getTextTrim(); 
      } 
     } 

SolrQuery query = new SolrQuery(); 
query.setQuery(qs); 


List facetList = doc.getRootElement().getChildren("facet"); 
        Iterator<String> it = facetList.iterator(); 
        while (it.hasNext()) { 
         Element el = (Element)it.next(); // 
         String name = el.getAttributeValue("name"); 
         String value = el.getTextTrim(); 
         if (name != null && value != null) {  
          facets.add(name+":"+value); 
         } 

        } 


query.setQuery(qs). 
      setFacet(Boolean.parseBoolean(_facetUse)). 
      setFacetMinCount(Integer.parseInt(_facetMinCount)). 
      setFacetLimit(Integer.parseInt(_facetLimit)). 

     for (int i=0; i<_facetFields.length; i++) { 
      query.addFacetField(_facetFields[i]);  
     }; 

     for (int i=0; i<facets.size(); i++) { 
      query.addFilterQuery(facets.get(i)); 
     }; 
    return query; 

    } 

我建议沿着这些线路使用的东西:

SolrQuery solrQuery = new SolrQuery(); 
    solrQuery.set(CommonParams.FQ, “-author:Dickens,Janet”); 

但是,这似乎是一种硬编码方法,它不能轻易应用于所有3个方面和所有方面值。我看过this,但是我仍然不清楚应该如何在当前代码中包含排除变体。你能帮忙吗?

感谢的确,

编辑1

我重视的代码来构建/准备Solr的查询,但我应该还包括Solr的实际情况是怎么样查询:

private QueryResponse execQuery(SolrQuery query) throws SolrServerException { 
    QueryResponse rsp = solrServer.query(query); 
    return rsp;  

} 

此外,发布转换Solr查询的代码会很有帮助响应的方面转化为可通过Web应用程序来理解:

Element elfacets = new Element("facets"); 
      List<FacetField> facets = rsp.getFacetFields(); 
      if (facets != null) { 
       int i = 0; 
       for (FacetField facet : facets) { 
        Element sfacet = new Element("facet"); 
        sfacet.setAttribute("name", facet.getName()); 

        List<Count> facetEntries = facet.getValues(); 

        for(FacetField.Count fcount : facetEntries) { 
         Element facetEntry = new Element("facetEntry"); 
         facetEntry.setText(fcount.getName()); 
         facetEntry.setAttribute("count", String.valueOf(fcount.getCount())); 
         sfacet.addContent(facetEntry); 
        } 
        elfacets.addContent(sfacet); 

      } 
      root.addContent(elfacets); 
     } 


     doc.addContent(root); 

     return doc; 
    } 

"facets"无非是XSLT,其中包含如何通过Web应用程序的处理,以Solr的面与面映射规则的更多。

编辑2 我附上"facets"模板,其通过该代码被称为在EDIT 1提出:

<xsl:template name="facets"> 
       <xsl:param name="q" /> 
       <xsl:analyze-string select="$q" regex='AND facet_(.*?):\(("?.*?"?)\)'> 
         <xsl:matching-substring> 
         <xsl:choose> 
         <xsl:when test="regex-group(1) = 'author'"> 
            <facet name="author"><xsl:value-of select="regex-group(2)" /></facet> 
         </xsl:when>  
         </xsl:choose> 
         </xsl:matching-substring> 
         <xsl:non-matching-substring> 
         <!--<xsl:analyze-string select="$q" regex='AND NOT facet_(.*?):\(("?.*?"?)\)'> 
         <xsl:matching-substring> 
         <xsl:choose> 
         <xsl:when test="regex-group(1) = 'author'"> 
            <facet name="author"><xsl:value-of select="regex-group(2)" /></facet> 
         </xsl:when> 
         </xsl:choose> 
        </xsl:matching-substring> 
        </xsl:analyze-string>--> 
        </xsl:non-matching-substring> 
       </xsl:analyze-string> 
    </xsl:template> 
</xsl:stylesheet> 

该模板只采用了author面,但是我有总共3个方面。应该注意的是,我的Web应用程序具有以下用于排除构面的语法:

AND NOT facet_author:("Dickens, Janet") 
+1

你想在需要的基础上排除'fq',对吗?你能控制参数传递给查询吗? – buddy86

+0

是的,那正是我想要做的!你如何将参数传递给查询?哪里?你能提供例子吗?请参阅我的附加编辑,其中显示模板构面。感谢您的帮助。 – paranza

回答

2

我确定您在某种方法中有以下几行。而不是硬编码的fq部分,有一些变数。

SolrQuery solrQuery = new SolrQuery(); 
solrQuery.set(CommonParams.FQ, “-author:Dickens,Janet”); 

如果您需要使用fq,请传递适当的参数(例如“-author:Dickens,Janet”)。否则传递一个空字符串。所以,你的查询将会像

/solr/solrbase/select?q=Dickens&fq=&wt=json&indent=true 

然后添加你的faceting部分的查询。虽然你的查询有fq=,但它不会引发错误。它基本上不会为fq部分工作。但其余的查询将正常工作。

希望这会有所帮助。

+0

对延迟回复表示歉意,但我已离开。我不能像你的例子那样传递一个合适的参数。你是否建议使用类似solrQuery.set(CommonParams.FQ)的东西?非常感谢! – paranza