2016-06-26 11 views
2

Gegeben Sammlung (#name: Benutzer) Struktur:Convert ISO Datum yyyy-mm-dd Format

{ 
"_id" : ObjectId("57653dcc533304a40ac504fc"), 
"username" : "XYZ", 
"followers" : [ 
    { 
     "count" : 31, 
     "ts" : ISODate("2016-06-17T18:30:00.996Z") 
    }, 
    { 
     "count" : 31, 
     "ts" : ISODate("2016-06-18T18:30:00.288Z") 
    } 
] 
} 

ich diese Sammlung abfragen möchten, basierend auf Benutzernamen Feld, und ts in ‚JJJJ- zurückgegeben werden MM-TT-Format. Erwartete Ausgabe:

{ 
"_id" : ObjectId("57653dcc533304a40ac504fc"), 
"username" : "XYZ", 
"followers" : [ 
    { 
     "count" : 31, 
     "date" : "2016-06-17" 
    }, 
    { 
     "count" : 31, 
     "date" : "2016-06-18" 
    } 
] 
} 

ich so etwas wie dies versucht haben:

db.users.aggregate([ 
{$match:{"username":"xyz"}}, 
{$project:{ "followers":{"count":1, 
     "date":"$followers.ts.toISOString().slice(0,10).replace(/-/g,'-')" 
      }} 
} 
]) 

Aber es scheint nicht zu funktionieren. Kann mir bitte jemand helfen? Vielen Dank.

Antwort

3

Betrachten Sie eine Aggregation Pipeline ausgeführt wird, der es Ihnen ermöglicht zunächst die Datenliste zu glätten, Projekt das neue Feld der $dateToString Operator, dann die abgeflachte docs umgruppieren das gewünschte Ergebnis zu erhalten.

Das Obige kann in drei verschiedenen Rohrleitungen gezeigt werden:

db.users.aggregate([ 
    { "$match": { "username": "xyz" } }, 
    { "$unwind": "$followers" }, 
    { 
     "$project": { 
      "username": 1, 
      "count": "$followers.count", 
      "date": { "$dateToString": { "format": "%Y-%m-%d", "date": "$followers.ts" } } 
     } 
    }, 
    { 
     "$group": { 
      "_id": "$_id", 
      "username": { "$first": "$username" }, 
      "followers": { "$push": { 
       "count": "$count", 
       "date": "$date" 
      }} 
     } 
    } 
]) 

Mit MongoDB 3.4 und höher, können Sie den neuen $addFields Pipeline Schritt zusammen Feld mit $map erstellen dem Array verwenden ohne die Notwendigkeit, abzuwickeln und zu gruppieren:

db.users.aggregate([ 
    { "$match": { "username": "xyz" } },  
    { 
     "$addFields": { 
      "followers": { 
       "$map": { 
        "input": "$followers", 
        "as": "follower", 
        "in": { 
         "count": "$$follower.count", 
         "date": { 
          "$dateToString": { 
           "format": "%Y-%m-%d", 
           "date": "$$follower.ts" 
          } 
         } 
        } 
       } 
      } 
     } 
    } 
]) 
+1

Dank Mann. :) Es hat perfekt funktioniert. – Veer

0

Der beste und einfachste Weg, dies zu tun, besteht darin, jedes Element im Array mit dem Operator $map zu transformieren. Natürlich müssen Sie im "in" -Ausdruck das $dateToString verwenden, um "Datum" in eine Zeichenkette unter Verwendung einer format specifiers umzuwandeln.

db.coll.aggregate(
    [ 
     { "$match": { "username": "XYZ" } }, 
     { "$project": { 
      "username": 1, 
      "followers": { 
       "$map": { 
        "input": "$followers", 
        "as": "f", 
        "in": { 
         "count": "$$f.count", 
         "date": { 
          "$dateToString": { 
           "format": "%Y-%m-%d", 
           "date": "$$f.ts" 
          } 
         } 
        } 
       } 
      } 
     }} 
    ] 
) 

, die produziert:

{ 
    "_id" : ObjectId("57653dcc533304a40ac504fc"), 
    "username" : "XYZ", 
    "followers" : [ 
     { 
      "count" : 31, 
      "date" : "2016-06-17" 
     }, 
     { 
      "count" : 31, 
      "date" : "2016-06-18" 
     } 
    ] 
}