/*
这道题目可以想到的是枚举一个参数,然后对于另一个参数做类似与最小生成树的东西,然而这样会超时,所以可以按照一个参数排序,然后进行删除或添加边的操作就行了,可以用LCT实现。
有一个地方要注意,LCT的边权是不能直接转化成点权的,所以我们可以把一条边想象成一个在两点之间的新点来做。
*/
#include<cstdio>
#include<algorithm>
#include<iostream>
#define N 200010
#define inf 1000000000
using namespace std;
int fa[N],son[N][2],val[N],mx[N],rev[N],st[N],p[N],n,m,ans=inf;
struct node{
int u,v,a,b;
};node e[N];
bool cmp(const node&s1,const node&s2){
return s1.a<s2.a;
}
int find(int x){
if(p[x]==x) return x;
return p[x]=find(p[x]);
}
bool isroot(int x){
return son[fa[x]][0]!=x&&son[fa[x]][1]!=x;
}
void pushdown(int x){
if(rev[x]){
swap(son[x][0],son[x][1]);
rev[son[x][0]]^=1;rev[son[x][1]]^=1;
rev[x]=0;
}
}
void pushup(int x){
int l=son[x][0],r=son[x][1];
mx[x]=x;
if(val[mx[l]]>val[mx[x]]) mx[x]=mx[l];
if(val[mx[r]]>val[mx[x]]) mx[x]=mx[r];
}
void rotate(int x){
int y=fa[x],z=fa[y],l,r;
if(son[y][0]==x)l=0;else l=1;r=l^1;
if(!isroot(y)){
if(son[z][0]==y)son[z][0]=x;
else son[z][1]=x;
}
fa[x]=z;fa[y]=x;fa[son[x][r]]=y;
son[y][l]=son[x][r];son[x][r]=y;
pushup(y);pushup(x);
}
void splay(int x){
int top=0;st[++top]=x;
for(int i=x;!isroot(i);i=fa[i])
st[++top]=fa[i];
for(int i=top;i;i--) pushdown(st[i]);
while(!isroot(x)){
int y=fa[x],z=fa[y];
if(!isroot(y)){
if(son[z][0]==y^son[y][0]==x) rotate(x);
else rotate(y);
}
rotate(x);
}
}
void access(int x){
int t=0;
while(x){
splay(x);
son[x][1]=t;
pushup(x);
t=x;x=fa[x];
}
}
void makeroot(int x){
access(x);
splay(x);
rev[x]^=1;
}
void join(int x,int y){
makeroot(x);
fa[x]=y;
}
void cut(int x,int y){
makeroot(x);
access(y);
splay(y);
fa[x]=son[y][0]=0;
pushup(y);
}
int query(int x,int y){
makeroot(x);access(y);splay(y);
return mx[y];
}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=m;i++)
scanf("%d%d%d%d",&e[i].u,&e[i].v,&e[i].a,&e[i].b);
sort(e+1,e+m+1,cmp);
for(int i=1;i<=n;i++) p[i]=i;
for(int i=1;i<=m;i++){
int u=e[i].u,v=e[i].v,a=e[i].a,b=e[i].b;
if(find(u)==find(v)){
int t=query(u,v);
if(val[t]>e[i].b){
cut(t,e[t-n].u);
cut(t,e[t-n].v);
}
else {
if(find(1)==find(n)) ans=min(ans,e[i].a+val[query(1,n)]);
continue;
}
}
else p[find(u)]=find(v);
val[i+n]=e[i].b;mx[i+n]=n+i;
join(i+n,e[i].u);join(i+n,e[i].v);
if(find(1)==find(n)) ans=min(ans,e[i].a+val[query(1,n)]);
}
if(find(1)==find(n))printf("%d",ans);
else printf("-1");
return 0;
}